BZOJ_2792_[Poi2012]Well_二分答案

时间:2022-02-24 11:34:15

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

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);
}