BZOJ_2792_[Poi2012]Well_二分答案
Description
给出n个正整数X1,X2,...Xn,可以进行不超过m次操作,每次操作选择一个非零的Xi,并将它减一。
最终要求存在某个k满足Xk=0,并且z=max{|Xi - Xi+1|}最小。
输出最小的z和此时最小的k。
Input
第一行两个正整数n, m (1<=n<=1,000,000, 1<=m<=10^18)。第二行n个正整数X1,X2,...Xn (Xi<=10^9)。
Output
输出k和z。数据保证方案一定存在。
Sample Input
16 15
8 7 6 5 5 5 5 5 6 6 7 8 9 7 5 5
8 7 6 5 5 5 5 5 6 6 7 8 9 7 5 5
Sample Output
1 2
HINT
将X序列变为
0 2 4 5 5 5 5 5 6 6 7 8 9 7 5 5
此时k=1,z=2,共操作了8+5+2=15次。
分析:首先肯定要二分答案出z,然后转化为判定是否能用小于等于m次操作使序列满足题意。
假设把第一个位置变成零,那么贪心的让后面比前面高度正好大z,扫一遍即可,可以发现最后的序列有若干个等差数列。
然后考虑怎么推出把其他位置i变成零对答案的贡献,i右边的数会下去一些,i左边的数会上去一些,可以发现这个满足单调性。
设f[i]为把第i个数变为0,左边需要改多少次,g[i]表示右边需要改多少次,双指针扫一遍,找到第一个大于|(j-i)|*z的位置,算出贡献即可。
有个小问题,可能序列一开始是不合法的,这是我们要从左往右再从右往左扫一遍,把不合法的部分削掉。
代码:
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 1000050
typedef long long ll;
int n,a[N],b[N],maxx;
ll m,s[N],f[N],g[N];
int check(int p) {
ll re=0;
int i,j;
for(i=1;i<=n;i++) b[i]=a[i];
for(int i=2;i<=n;i++)
if(b[i]-b[i-1]>p)
re+=b[i]-b[i-1]-p,b[i]=b[i-1]+p;
for(int i=n-1;i;i--)
if(b[i]-b[i+1]>p)
re+=b[i]-b[i+1]-p,b[i]=b[i+1]+p;
if(re>m) return 0;
for(i=1;i<=n;i++) s[i]=s[i-1]+b[i];
for(i=1,j=1;i<=n;i++) {
while(j<i&&b[j]<=1ll*(i-j)*p) j++;
f[i]=s[i-1]-s[j-1]-1ll*(i-j)*(i-j+1)/2*p;
}
for(i=n,j=n;i;i--) {
while(j>i&&b[j]<=1ll*(j-i)*p) j--;
g[i]=s[j]-s[i]-1ll*(j-i)*(j-i+1)/2*p;
}
for(i=1;i<=n;i++) if(f[i]+g[i]+b[i]+re<=m) return i;
return 0;
}
int main() {
scanf("%d%lld",&n,&m);
int i;
for(i=1;i<=n;i++) {
scanf("%d",&a[i]);
maxx=max(maxx,a[i]);
}
int l=0,r=maxx+1;
while(l<r) {
int mid=(l+r)>>1;
if(check(mid)) r=mid;
else l=mid+1;
}
printf("%d %d\n",check(l),l);
}