2019ICPC南昌邀请赛网络赛 I. Max answer (单调栈+线段树/笛卡尔树)

题目链接

题意:求一个序列的最大的(区间最小值*区间和)

线段树做法:用单调栈求出每个数两边比它大的左右边界,然后用线段树求出每段区间的和sum、最小前缀lsum、最小后缀rsum,枚举每个数a[i],设以a[i]为最小值的区间为[l,r]

若a[i]>0,则最优解就是a[i]*([l,r]的区间和),因为[l,r]上的数都比a[i]大。

若a[i]<0,则最优解是a[i]*([l,i-1]上的最小后缀+a[i]+[i+1,r]上的最小前缀),在线段树上查询即可。

复杂度$O(nlogn)$

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 const int N=5e5+10,inf=0x3f3f3f3f;
 5 int a[N],n,sta[N],L[N],R[N],tp;
 6 #define ls (u<<1)
 7 #define rs (u<<1|1)
 8 #define mid ((l+r)>>1)
 9 struct D {ll sum,lsum,rsum;} s[N<<2];
10 D mg(D a,D b) {
11     D t= {0,0};
12     t.sum=a.sum+b.sum;
13     t.lsum=min(a.lsum,a.sum+b.lsum);
14     t.rsum=min(b.rsum,b.sum+a.rsum);
15     return t;
16 }
17 void build(int u=1,int l=1,int r=n) {
18     if(l==r) {s[u].sum=a[l],s[u].lsum=min((ll)a[l],0ll),s[u].rsum=min((ll)a[l],0ll); return;}
19     build(ls,l,mid),build(rs,mid+1,r),s[u]=mg(s[ls],s[rs]);
20 }
21 void qry(int L,int R,D& x,int u=1,int l=1,int r=n) {
22     if(l>=L&&r<=R) {x=mg(x,s[u]); return;}
23     if(l>R||r<L)return;
24     qry(L,R,x,ls,l,mid),qry(L,R,x,rs,mid+1,r);
25 }
26 int main() {
27     scanf("%d",&n);
28     a[0]=a[n+1]=~inf;
29     for(int i=1; i<=n; ++i)scanf("%d",&a[i]);
30     sta[tp=0]=0;
31     for(int i=1; i<=n; ++i) {
32         for(; a[sta[tp]]>=a[i]; --tp);
33         L[i]=sta[tp]+1,sta[++tp]=i;
34     }
35     sta[tp=0]=n+1;
36     for(int i=n; i>=1; --i) {
37         for(; a[sta[tp]]>=a[i]; --tp);
38         R[i]=sta[tp]-1,sta[++tp]=i;
39     }
40     build();
41     ll ans=0;
42     for(int i=1; i<=n; ++i) {
43         if(a[i]>0) {
44             D t= {0,0};
45             qry(L[i],R[i],t);
46             ans=max(ans,a[i]*t.sum);
47         } else if(a[i]<0) {
48             ll x=0;
49             D t= {0,0};
50             qry(L[i],i,t);
51             x+=t.rsum;
52             t= {0,0};
53             qry(i,R[i],t);
54             x+=t.lsum;
55             x-=a[i];
56             ans=max(ans,a[i]*x);
57         }
58     }
59     printf("%lld\n",ans);
60     return 0;
61 }

笛卡尔树做法:对整个序列建立笛卡尔树,用和线段树相同的方法求出每个结点的子树所代表区间的sum,lsum,rsum,枚举每个结点,如果是正数则乘上该结点的sum,如果是负数则乘上该结点的(左儿子的rsum+右儿子的lsum+结点本身的值)即可。

复杂度$O(n)$

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 typedef long long ll;
 4 const int N=5e5+10,inf=0x3f3f3f3f;
 5 int n,a[N],ls[N],rs[N],sta[N],tp;
 6 ll sum[N],lsum[N],rsum[N];
 7 void build() {
 8     a[n+1]=~inf,sta[tp=0]=n+1;
 9     for(int i=1; i<=n; ++i) {
10         for(; a[i]<a[sta[tp]]; --tp);
11         ls[i]=rs[sta[tp]],rs[sta[tp]]=i,sta[++tp]=i;
12     }
13 }
14 void dfs(int u) {
15     if(!u)return;
16     dfs(ls[u]),dfs(rs[u]);
17     sum[u]=sum[ls[u]]+a[u]+sum[rs[u]];
18     lsum[u]=min(lsum[ls[u]],sum[ls[u]]+a[u]+lsum[rs[u]]);
19     rsum[u]=min(rsum[rs[u]],sum[rs[u]]+a[u]+rsum[ls[u]]);
20 }
21 int main() {
22     scanf("%d",&n);
23     for(int i=1; i<=n; ++i)scanf("%d",&a[i]);
24     build(),dfs(rs[n+1]);
25     ll ans=0;
26     for(int i=1; i<=n; ++i) {
27         if(a[i]>0)ans=max(ans,a[i]*sum[i]);
28         else if(a[i]<0)ans=max(ans,a[i]*(rsum[ls[i]]+a[i]+lsum[rs[i]]));
29     }
30     printf("%lld\n",ans);
31     return 0;
32 }

 

posted @ 2019-04-23 19:02  jrltx  阅读(526)  评论(0编辑  收藏  举报