P2048 [NOI2010]超級鋼琴

無關:忽然感受以前的碼風調代碼不太方便,仍是稍微改一下吧,畢竟如今競賽的代碼寫出來仍是要給本身看的...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 }
相關文章
相關標籤/搜索