RMQ+Heap+区间分裂。
对于任意区间右端点i,其左端点取值在l,r之间,若左端点为m,则v为max(sum[i]-sum[m-1]),显然这里i是不变的,所以可以用rmq查询m的位置,然后计算v。
现将所有右端点扫一遍,然后扔到堆里面,堆中节点记录的是决策,即右端点i,左端点区间,优先级由v决定。
然后取出堆顶,v加到ans里去,分裂[l,r]为[l,m-1]和[m+1,r],rmq出新的v,再将分裂后的区间加到堆里。
重复k次就好了。
#include<iostream> #include<cstdio> #include<cstring> #include<queue> using namespace std; const int N=500000+5; typedef long long ll; struct Heapnode{ int i,l,r,v,m; bool operator<(const Heapnode &rhs)const{ return v<rhs.v; } }; int n,k,L,R,st[N][21],sum[N]; void rmq_init(){ for(int i=0;i<=n;i++)st[i][0]=i; for(int j=1;(1<<j)<=n;j++) for(int i=0;i+(1<<j)-1<=n;i++){ int l=st[i][j-1],r=st[i+(1<<(j-1))][j-1]; if(sum[l]<sum[r])st[i][j]=l; else st[i][j]=r; } } int rmq(int l,int r){ l--;r--; int k=0; while((1<<(k+1))<=r-l+1)k++; return sum[st[l][k]]<sum[st[r-(1<<k)+1][k]]?st[l][k]:st[r-(1<<k)+1][k]; } int main(){ scanf("%d%d%d%d",&n,&k,&L,&R); for(int i=1;i<=n;i++)scanf("%d",&sum[i]); for(int i=1;i<=n;i++)sum[i]+=sum[i-1]; rmq_init(); priority_queue<Heapnode>q; ll ans=0; int l,r,v,m,i; for(i=1;i<=n;i++){ l=i-R+1;r=i-L+1; l=max(l,1);r=max(r,1); if(i-l+1<L)continue; m=rmq(l,r);m++;v=sum[i]-sum[m-1]; q.push((Heapnode){i,l,r,v,m}); } while(k--){ Heapnode x=q.top();q.pop(); ans+=x.v; i=x.i;l=x.l;r=x.r;m=x.m; int tmp; if(l<m){ tmp=rmq(l,m-1);tmp++; v=sum[i]-sum[tmp-1]; q.push((Heapnode){i,l,m-1,v,tmp}); } if(m<r){ tmp=rmq(m+1,r);tmp++; v=sum[i]-sum[tmp-1]; q.push((Heapnode){i,m+1,r,v,tmp}); } } printf("%lld",ans); return 0; }