• River Locks


     

     

    传送门

    题意:

    n个船闸,每个有体积为v, 每个船闸都有一个水管,水管每秒可以放1单位的水,当第i个船闸的水满时,会向i + 1船闸转移,转移的时间为0,有q次询问,每次询问一个时间t, 问在时间t下需要的最少的水管数使所有的船闸都装满水,不行的输出 -1

    思路:

    首先,贪心要选水管肯定是前面的优先,考虑前i个船闸装满水需要的最少时间,dp[i], 二分求得,最后询问,只要t >= dp[n] && count <= n就是可以的, count == (sum[n] + t - 1) / t(向上取整)

    1. #include
    2. #define endl '\n'
    3. #define IOS ios::sync_with_stdio(false);
    4. using namespace std;
    5. typedef long long ll;
    6. const int N = 2e5 + 10;
    7. ll n, t, q;
    8. ll v[N], dp[N]; //dp[i]代表将前i个船闸填满水需要的最少时间
    9. ll sum[N];
    10. int main()
    11. {
    12. IOS; cin.tie(0), cout.tie(0);
    13. cin >> n;
    14. for (int i = 1; i <= n; ++i)
    15. {
    16. cin >> v[i];
    17. }
    18. dp[1] = v[1], sum[1] = v[1];
    19. for (int i = 2; i <= n; ++i)
    20. {
    21. sum[i] = sum[i - 1] + v[i];
    22. ll l = dp[i - 1], r = 1e9, mid, ans = 1e9; //因为第i个填满需要的最少时间肯定是在前i - 1个转移过来的,所以l = dp[i - 1]
    23. while (l <= r)
    24. {
    25. mid = (l + r) >> 1;
    26. if (mid * i - sum[i - 1] >= v[i])
    27. {
    28. ans = min(ans, mid);
    29. r = mid - 1;
    30. }
    31. else
    32. l = mid + 1;
    33. }
    34. dp[i] = ans;
    35. }
    36. cin >> q;
    37. while (q--)
    38. {
    39. cin >> t;
    40. //需要的时间肯定要大于dp[n], 再次基础上sum[n] / t, 向上取整就是需要的最少水管数
    41. int count = (sum[n] + t - 1) / t;
    42. if (dp[n] <= t && count <= n)
    43. cout << count << endl;
    44. else
    45. cout << "-1" << endl;
    46. }
    47. return 0;
    48. }

  • 相关阅读:
    ArcGIS的地理空间大数据的数据分析图
    Docker-Windows安装使用
    安理【2022】
    解决el-form部分字段在输入的时候被带着走的问题
    【黄色手套22】12番外:图形库
    SSL证书对网站SEO的好处
    ubuntu设置定时开关机
    做亚马逊测评有哪些需要注意的?
    Dynamic Kafka Configurations
    【DevOps】Git 图文详解(一):简介及基础概念
  • 原文地址:https://blog.csdn.net/qq_63010655/article/details/125890597