题面
英文题面
题意:
笛卡尔树是一种既满足堆的性质,又满足二叉搜索树的性质的树。可以发现的是,对于一个排列,它的笛卡尔树是唯一的。
\(n \leq 2\times 10^5\)。
题解:发现笛卡尔树中的一个节点的权值可以表示为\(r_i-l_i+1\)的形式。其中\(r_i\)表示最小的\(p \geq i\),使得\(a_{p+1}>a_i\)。所以我们可以分别维护\(l_i\)和\(r_i\)。
考虑加入一个权值最大的点后,\(r_i\)和\(l_i\)会出现哪些变化。设当前插入的是数字\(i\),它在排列中的位置是\(p\),它在当前的由1到\(i\)组成的子序列中的位置为\(q\)。
- 对于\(\forall_{k
,\(r_k=min(r_k,q-1)\);
- 对于\(p\),\(l_p=1,r_p=i\)。
- 对于\(\forall_{k>p},l_k++,r_k++,l_k=max(l_k,q+1)\)。
所以相当于是区间取min,区间加和单点赋值。我们可以采用势能线段树的思想,维护:
- 区间和;
- 最大(小)值;
- 最大(小)值出现次数;
- 次大(小)值;
- 区间合法点个数;
- 加法标记和赋值标记。
对于区间取min操作,我们递归到一个区间,其最大值小于\(w\),次大值大于等于\(w\)时进行修改,否则继续递归子树。
不会势能线段树区间取min复杂度证明的可以出门右转自行百度。
注意pushup和pushdown各要分三种情况讨论,还要注意标记的下传顺序问题。
时间复杂度:\(O(nlog^2n)\)。
代码:
#include
using namespace std;
#define re register int
#define F(x,y,z) for(re x=y;x<=z;x++)
#define FOR(x,y,z) for(re x=y;x>=z;x--)
typedef long long ll;
#define I inline void
#define IN inline int
#define C(x,y) memset(x,y,sizeof(x))
#define STS system("pause")
templateI read(D &res){
res=0;register D g=1;register char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')g=-1;
ch=getchar();
}
while(isdigit(ch)){
res=(res<<3)+(res<<1)+(ch^48);
ch=getchar();
}
res*=g;
}
const int INF=1e9+7;
int n,m,a[202000],p[202000],t[202000];
IN lbt(int x){return x&(-x);}
I modi(int x,int w){while(x<=n)t[x]+=w,x+=lbt(x);}
IN ques(int x){re res=0;while(x)res+=t[x],x-=lbt(x);return res;}
#define all 1,1,n
#define lt k<<1,l,mid
#define rt k<<1|1,mid+1,r
ll sr[1606000],sl[1606000];int cntr[1606000],cntl[1606000];
int mxr[1606000],mxl[1606000],smxr[1606000],smxl[1606000],cmxr[1606000],cmxl[1606000],snl[1606000],snr[1606000];
int lazr[1606000],lazl[1606000],lmr[1606000],lml[1606000];
I build(int k,int l,int r){
sl[k]=sr[k]=cntr[k]=cntl[k]=cmxr[k]=cmxl[k]=lazr[k]=lazl[k]=lmr[k]=lml[k]=0;
mxr[k]=smxr[k]=-INF;mxl[k]=smxl[k]=INF;
if(l==r)return;
re mid=(l+r)>>1;
build(lt);build(rt);
}
I push_upl(int k){
sl[k]=sl[k<<1]+sl[k<<1|1];cntl[k]=cntl[k<<1]+cntl[k<<1|1];
if(mxl[k<<1]mxl[k<<1|1]){
snl[k]=2;
mxl[k]=mxl[k<<1|1];cmxl[k]=cmxl[k<<1|1];smxl[k]=min(smxl[k<<1|1],mxl[k<<1]);
}
else{
snl[k]=3;
mxl[k]=mxl[k<<1];cmxl[k]=cmxl[k<<1]+cmxl[k<<1|1];smxl[k]=min(smxl[k<<1],smxl[k<<1|1]);
}
}
I push_upr(int k){
sr[k]=sr[k<<1]+sr[k<<1|1];cntr[k]=cntr[k<<1]+cntr[k<<1|1];
if(mxr[k<<1]>mxr[k<<1|1]){
snr[k]=1;
mxr[k]=mxr[k<<1];cmxr[k]=cmxr[k<<1];smxr[k]=max(smxr[k<<1],mxr[k<<1|1]);
}
else if(mxr[k<<1]>1;
if(x<=mid)modi_l(lt,x,w);
else modi_l(rt,x,w);
push_upl(k);
}
I modi_r(int k,int l,int r,int x,int w){
if(l==r){
sr[k]=mxr[k]=w;smxr[k]=-INF;cmxr[k]=cntr[k]=1;
return;
}
push_downr(k);
re mid=(l+r)>>1;
if(x<=mid)modi_r(lt,x,w);
else modi_r(rt,x,w);
push_upr(k);
}
I revi_l(int k,int l,int r,int x,int y,int w){
if(x>r||y>1;
revi_l(lt,x,y,w);revi_l(rt,x,y,w);
push_upl(k);
}
I revi_r(int k,int l,int r,int x,int y,int w){
if(x>r||y>1;
revi_r(lt,x,y,w);revi_r(rt,x,y,w);
push_upr(k);
}
I fill_max(int k,int l,int r,int x,int y,int w){
if(x>r||y=w)return;
if(x<=l&&r<=y){
if(smxl[k]>=w){
// cout<<"#"<>1;
fill_max(lt,x,y,w);fill_max(rt,x,y,w);
return push_upl(k),void();
}
push_downl(k);
re mid=(l+r)>>1;
fill_max(lt,x,y,w);fill_max(rt,x,y,w);
push_upl(k);
}
I fill_min(int k,int l,int r,int x,int y,int w){
if(x>r||y>1;
fill_min(lt,x,y,w);fill_min(rt,x,y,w);
return push_upr(k),void();
}
push_downr(k);
re mid=(l+r)>>1;
fill_min(lt,x,y,w);fill_min(rt,x,y,w);
push_upr(k);
}
I getit(int k,int l,int r){
cout<<"!"<>1;
getit(lt);getit(rt);
}
int main(){
// freopen("a.out","w",stdout);
read(n);F(i,1,n)read(a[i]),p[a[i]]=i;build(all);
F(i,1,n){
modi_l(all,p[i],1),modi_r(all,p[i],i);
m=ques(p[i])+1;modi(p[i],1);revi_l(all,p[i]+1,n,1);revi_r(all,p[i]+1,n,1);
fill_max(all,p[i]+1,n,m+1);fill_min(all,1,p[i]-1,m-1);
printf("%lld\n",sr[1]-sl[1]+i);
// getit(all);
}
return 0;
}
/*
5
2 4 1 5 3
6
1 2 4 5 6 3
10
8 10 9 3 7 5 4 6 2 1
15
1 7 12 14 15 9 8 10 11 2 5 4 3 13 6
*/