题意

在一个数轴上有n个集装箱,第 i 个集装箱的位置为x[i],且在集装箱内装有a[i]件货物,现在将这些集装箱内的货物进行移动(将一件货物从第 i 个集装箱移动到第 j 个集装箱的花费就为2*abs(x[i]-x[j]) ),求在总花费不超过T的情况下,最多能将多少货物移动到同一个集装箱内。

分析

既然要使得花费在不超过T的情况尽可能多的移动货物,那么我们肯定是将一个区间内的所有货物移到坐标中位的集装箱上。那么我们就可以对答案进行二分,然后枚举所要移动的区间的左端点,再找到中位点和右端点,然后判断这个区间移动的花费是否小于T。

可以预处理一下前缀和以及前缀花费、距离等。依次来计算区间的花费。

二分判断的过程中,由于最终的答案不一定是将整个区间内所有的货物都移动到一个集装箱,所以我们还要判断比需求量多的那部分是从左端点移过来的还剩从右端点移过来的,然后再根据花费的大小情况进行二分就可以得到最终答案了。

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cmath>
#include<string>
#include<cstring>
using namespace std;
#define ll long long
const int N = 5e5 + ;
ll dis[N]; ///dis[i]表示第i个集装箱距离x=0的距离
ll num[N]; ///num[i]表示第i个集装箱有的产品个数
ll sum[N]; ///sum[i]表示第1~i个集装箱有的产品数总和
ll cost[N]; ///cost[i]表示第1~i个集装箱的所有产品移到x=0处所"获得"的费用
ll tot,q;
int n, L, R, u; ///u为产品汇总的最优点
ll cost_l(int i) {///费用=将L+1~i中所有产品移动到i处所需费用+(将i~R中所有的产品移动到i处所需的费用 - 区间L+1~R比x多的产品数从R处移动到i处的费用)
return ((sum[i] - sum[L])*dis[i] - (cost[i] - cost[L])) + ((cost[R] - cost[i]) - (sum[R] - sum[i])*dis[i] - (sum[R] - sum[L] - q)*(dis[R] - dis[i]));
}
ll cost_r(int i) {///同上,只是规则是优先从右边界开始取产品
return -((sum[R] - sum[i])*dis[i] - (cost[R] - cost[i])) - ((cost[i-] - cost[L]) - (sum[i-] - sum[L])*dis[i] + (sum[R] - sum[L] - q)*(dis[i] - dis[L+]));
}
bool check(ll &x) { ///判断需求x是否能在所给的费用t内达到
q = x;
L = , R = , u = ;
while () { ///从左边界开始向右移动区间,优先取区间左边的产品
while (R < n&&sum[R] - sum[L] < x) R++;
if (sum[R] - sum[L] < x)break; ///若是当前的L~n无法满足x,那么L++也不可能满足
while (u < L)u++;
while (u < R&&cost_l(u)>cost_l(u + ))u++;
if (cost_l(u) <= tot)return true;
L++;
}
L = n - , R = n, u = n;
while () { ///从右边界开始向左移动区间,优先取区间右边的产品
while (L > && sum[R] - sum[L] < x) L--;
if (sum[R] - sum[L] < x)break; ///若是当前的L~R无法满足x,那么R--也不可能满足
while (u > R)u--;
while (u > L && cost_r(u) > cost_r(u - ))u--;
if (cost_r(u) <= tot)return true;
R--;
}
return false;
}
int main()
{
scanf("%d%lld", &n, &tot);
tot /= ;
for (int i = ; i <= n; i++)
scanf("%lld", &dis[i]);
sum[]=cost[]=;
for (int i = ; i <= n; i++) {
scanf("%lld", &num[i]);
sum[i] = sum[i - ] + num[i];
cost[i] = cost[i - ] + num[i] * dis[i];
}
ll l = ,r = sum[n] + ;
while (l + < r) {
ll mid = (l + r) >> ;
if (check(mid)) l = mid;
else r = mid;
}
printf("%lld\n", l);
return ;
}

2018牛客网暑期ACM多校训练营(第二场)G Transform(二分)的更多相关文章

  1. 2018牛客网暑期ACM多校训练营(第二场)I- car ( 思维)

    2018牛客网暑期ACM多校训练营(第二场)I- car 链接:https://ac.nowcoder.com/acm/contest/140/I来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 ...

  2. 2018牛客网暑期ACM多校训练营(第一场)D图同构,J

    链接:https://www.nowcoder.com/acm/contest/139/D来源:牛客网 同构图:假设G=(V,E)和G1=(V1,E1)是两个图,如果存在一个双射m:V→V1,使得对所 ...

  3. 2018 牛客网暑期ACM多校训练营(第一场) E Removal (DP)

    Removal 链接:https://ac.nowcoder.com/acm/contest/139/E来源:牛客网 题目描述 Bobo has a sequence of integers s1, ...

  4. 2018牛客网暑期ACM多校训练营(第十场)A Rikka with Lowbit (树状数组)

    链接:https://ac.nowcoder.com/acm/contest/148/A 来源:牛客网 Rikka with Lowbit 时间限制:C/C++ 5秒,其他语言10秒 空间限制:C/C ...

  5. 2018牛客网暑期ACM多校训练营(第十场)J Rikka with Nickname(二分,字符串)

    链接:https://ac.nowcoder.com/acm/contest/148/J?&headNav=acm 来源:牛客网 Rikka with Nickname 时间限制:C/C++ ...

  6. 2018牛客网暑期ACM多校训练营(第二场)J Farm(树状数组)

    题意 n*m的农场有若干种不同种类作物,如果作物接受了不同种类的肥料就会枯萎.现在进行t次施肥,每次对一个矩形区域施某种类的肥料.问最后枯萎的作物是多少. 分析 作者:xseventh链接:https ...

  7. 2018牛客网暑期ACM多校训练营(第一场)B Symmetric Matrix(思维+数列递推)

    题意 给出一个矩阵,矩阵每行的和必须为2,且是一个主对称矩阵.问你大小为n的这样的合法矩阵有多少个. 分析 作者:美食不可负064链接:https://www.nowcoder.com/discuss ...

  8. 2018牛客网暑期ACM多校训练营(第三场) A - PACM Team - [四维01背包][四约束01背包]

    题目链接:https://www.nowcoder.com/acm/contest/141/A 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...

  9. 2018牛客网暑期ACM多校训练营(第五场) F - take - [数学期望][树状数组]

    题目链接:https://www.nowcoder.com/acm/contest/143/F 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...

  10. 2018牛客网暑期ACM多校训练营(第五场) E - room - [最小费用最大流模板题]

    题目链接:https://www.nowcoder.com/acm/contest/143/E 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K ...

随机推荐

  1. 【XSY2469】graph 分治 并查集

    题目大意 给你一张\(n\)个点\(m\)条边的无向图,问删去每个点后,原图是不是二分图. \(n,m\leq 100000\) 题解 一个图是二分图\(\Longleftrightarrow\)该图 ...

  2. git 出现stderr: error: bad signature fatal: index file corrupt

    命令执行依次: $ rm -f .git/index $ git reset 重启即可

  3. MT【258】椭圆第三定义

    如图,已知椭圆方程为$\dfrac{x^2}{4}+\dfrac{y^2}{3}=1$$A$为椭圆上一点,$AF_1,AF_2$与椭圆交于$B,C$两点,$A_1B,A_2C$交于一点$M$.当$A$ ...

  4. 【题解】 bzoj3916: [Baltic2014]friends (字符串Hash)

    题面戳我 Solution 首先长度为偶数可以直接判掉 然后我们可以枚举删的位置,通过预处理的\(hash\),判断剩余部分是否划分成两个一样的 判重要注意,我们把字符串分为三个部分\(L_l+1+L ...

  5. BZOJ 4242: 水壶(Kruskal重构树 + Bfs)

    题意 一块 \(h ∗ w\) 的区域,存在障碍.空地.\(n\) 个建筑,从一个建筑到另一个建筑的花费为:路径上最长的连续空地的长度. \(q\) 次询问:从建筑 \(s_i\) 到 \(t_i\) ...

  6. 【BZOJ3174】[TJOI2013]拯救小矮人(贪心,动态规划)

    [BZOJ3174][TJOI2013]拯救小矮人(贪心,动态规划) 题面 BZOJ 洛谷 题解 我们定义一个小矮人的\(A_i+B_i\)为它的逃跑能力. 我们发现,如果有两个小矮人\(x,y\), ...

  7. 「JLOI2015」管道连接 解题报告

    「JLOI2015」管道连接 先按照斯坦纳树求一个 然后合并成斯坦纳森林 直接枚举树的集合再dp一下就好了 Code: #include <cstdio> #include <cct ...

  8. Apache rewrite地址重写

    Apache-rewrite+13个经典案例Apache 重写规则的常见应用(rewrite)一:目的 如何用Apache重写规则来解决一些常见的URL重写方法的问题,通过常见的 实例给用户一些使用重 ...

  9. POJ--2104 K-th Number (主席树模版题)

    题目链接 求区间第k大 #include<iostream> #include<cstring> #include<algorithm> #include<v ...

  10. Docker自动补全容器名

    Zsh Place the completion script in your /path/to/zsh/completion (typically ~/.zsh/completion/): 下载自动 ...