River Locks

1 篇文章 0 订阅

 

 

传送门

题意:

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(向上取整)

#include <bits/stdc++.h>
#define endl '\n'
#define IOS ios::sync_with_stdio(false);
using namespace std;

typedef long long ll;
const int N = 2e5 + 10;
ll n, t, q;
ll v[N], dp[N]; //dp[i]代表将前i个船闸填满水需要的最少时间
ll sum[N];

int main()
{
    IOS; cin.tie(0), cout.tie(0);
    cin >> n;
    for (int i = 1; i <= n; ++i)
    {
        cin >> v[i];
    }

    dp[1] = v[1], sum[1] = v[1];
    for (int i = 2; i <= n; ++i)
    {
        sum[i] = sum[i - 1] + v[i];
        ll l = dp[i - 1], r = 1e9, mid, ans = 1e9;    //因为第i个填满需要的最少时间肯定是在前i - 1个转移过来的,所以l = dp[i - 1]
        while (l <= r)
        {
            mid = (l + r) >> 1;
            if (mid * i - sum[i - 1] >= v[i])
            {
                ans = min(ans, mid);
                r = mid - 1;
            }
            else
                l = mid + 1;
        }
        dp[i] = ans;
    }

    cin >> q;
    while (q--)
    {
        cin >> t;
        //需要的时间肯定要大于dp[n], 再次基础上sum[n] / t, 向上取整就是需要的最少水管数
        int count = (sum[n] + t - 1) / t;
        if (dp[n] <= t && count <= n)
            cout << count << endl;
        else
            cout << "-1" << endl;
    }
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值