無關:忽然感受以前的碼風調代碼不太方便,仍是稍微改一下吧,畢竟如今競賽的代碼寫出來仍是要給本身看的...c++
這道題,顯然肯定了一個端點以後這個端點對應的區間和的最大值是定的,很容易想到用ST表維護一個前綴和最大值git
而後再維護一個大根堆,記錄每一個合法最大值區間spa
在每個狀態時,堆頂元素必定是最優解,因此每次取堆頂元素便可code
可是把堆頂元素取出來以後,以後最優解還有可能在當前堆頂的集合中的子集裏產生,因此還須要把當前堆頂有可能產生的其餘貢獻丟到堆裏blog
設堆頂元素在位置pos處取到最優解的狀況,那麼由於兩個和絃不能相同,可能的新的貢獻的右端點只有可能在$[l,pos-1]$和$[pos+1,r]$中取到get
把這兩個區間也丟進去維護就行了it
1 //hgs AK IOI,IMO,ICHO,IPHO 2 #include<bits/stdc++.h> 3 #define int long long 4 #define writeln(x) write(x),puts("") 5 #define writep(x) write(x),putchar(' ') 6 using namespace std; 7 inline int read(){ 8 int ans=0,f=1;char chr=getchar(); 9 while(!isdigit(chr)){if(chr=='-') f=-1;chr=getchar();} 10 while(isdigit(chr)){ans=(ans<<3)+(ans<<1)+chr-48;chr=getchar();} 11 return ans*f; 12 }void write(int x){ 13 if(x<0) putchar('-'),x=-x; 14 if(x>9) write(x/10); 15 putchar(x%10+'0'); 16 }const int M = 5E5+5; 17 int a[M],s[M],st[M][21],n,m,sl,sr,lg2[M]={-1},bin[21]={1}; 18 inline int max(int x,int y){return ((x>y)?(x):(y));} 19 inline int min(int x,int y){return ((x>y)?(y):(x));} 20 inline int chkmax(int x,int y){return (s[x]>s[y])?(x):(y);} 21 inline void Pre() 22 { 23 for(int i=1;i<=20;i++) 24 bin[i]=bin[i-1]<<1; 25 for(int i=1;i<=n;i++) 26 lg2[i]=lg2[i>>1]+1, 27 st[i][0]=i; 28 for(int j=1;j<=20;j++) 29 for(int i=1;i+bin[j]-1<=n;i++) 30 st[i][j]=chkmax(st[i][j-1],st[i+bin[j-1]][j-1]); 31 } 32 inline int Query(int l,int r) 33 { 34 return chkmax(st[l][lg2[r-l+1]],st[r-bin[lg2[r-l+1]]+1][lg2[r-l+1]]); 35 } 36 struct P 37 { 38 int x,l,r; 39 inline int Calc()const 40 { 41 return s[Query(l,r)]-s[x-1]; 42 } 43 friend bool operator<(const P&a,const P&b) 44 { 45 return a.Calc()<b.Calc(); 46 } 47 }; 48 priority_queue<P>q; 49 inline P Make(int x,int l,int r) 50 { 51 r=min(r,n); 52 return (P){x,l,r}; 53 } 54 inline void Split(const P&x) 55 { 56 P t1,t2; 57 int pos=Query(x.l,x.r); 58 if(x.l!=pos)q.push(Make(x.x,x.l,pos-1)); 59 if(x.r!=pos)q.push(Make(x.x,pos+1,x.r)); 60 } 61 signed main() 62 { 63 n=read(),m=read(),sl=read(),sr=read(); 64 for(int i=1,x;i<=n;i++) 65 s[i]=read(),s[i]+=s[i-1]; 66 Pre(); 67 for(int i=1;i+sl-1<=n;i++) 68 q.push(Make(i,i+sl-1,i+sr-1)); 69 register int ans(0); 70 while(m--) 71 { 72 P now=q.top(); 73 q.pop(); 74 ans+=now.Calc(); 75 Split(now); 76 }cout<<ans; 77 return 0; 78 }