标签:
#include <ctime>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 100010
using namespace std;
int size,n,root;
struct node
{
int l,r,rnd,siz,w,v;
}tr[N];
void pushup(int rt)
{
tr[rt].siz=tr[tr[rt].l].siz+tr[tr[rt].r].siz+1;
}
void lturn(int &rt)
{
int t=tr[rt].r;
tr[rt].r=tr[t].l;
tr[t].l=rt;
tr[t].siz=tr[rt].siz;
pushup(rt);
rt=t;
}
void rturn(int &rt)
{
int t=tr[rt].l;
tr[rt].l=tr[t].r;
tr[t].r=rt;
tr[t].siz=tr[rt].siz;
pushup(rt);
rt=t;
}
void insert(int &rt,int v)
{
if(!rt)
{
rt=++size;
tr[rt].l=tr[rt].r=0,tr[rt].siz=1;
tr[rt].v=size,tr[rt].rnd=rand();
return;
}
tr[rt].siz++;
if(v<=tr[tr[rt].l].siz+1)
{
insert(tr[rt].l,v);
if(tr[tr[rt].l].rnd<tr[rt].rnd)rturn(rt);
}else
{
insert(tr[rt].r,v-tr[tr[rt].l].siz-1);
if(tr[tr[rt].r].rnd<tr[rt].rnd)lturn(rt);
}
}
int tot;
int seq[N];
void getseq(int rt)
{
if(tr[rt].l!=0)
getseq(tr[rt].l);
seq[++tot]=tr[rt].v;
if(tr[rt].r!=0)
getseq(tr[rt].r);
}
int f[N],d[N],ans[N];
int print[N];
void getlis()
{
memset(d,0x3f,sizeof(d));
d[0]=0;
d[1]=seq[1],f[1]=1;
int mx=1;
for(int i=2;i<=n;i++)
{
if(seq[i]>d[mx])
{
f[i]=++mx;
d[mx]=min(d[mx],seq[i]);
}else
{
int l=0,r=mx,ans=0;
while(l<=r)
{
int mid=(l+r)>>1;
if(d[mid]<seq[i])ans=mid,l=mid+1;
else r=mid-1;
}
f[i]=ans+1;
d[ans+1]=min(d[ans+1],seq[i]);
}
}
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
int x;
scanf("%d",&x);x++;
insert(root,x);
}
getseq(root);
getlis();
for(int i=1;i<=n;i++)print[seq[i]]=f[i];
for(int i=1;i<=n;i++)print[i]=max(print[i],print[i-1]);
for(int i=1;i<=n;i++)printf("%d\n",print[i]);
}
版权声明:本文为博主原创文章,未经博主允许不得转载。
BZOJ 3173 [Tjoi2013]最长上升子序列 Treap+LIS
标签:
原文地址:http://blog.csdn.net/wzq_qwq/article/details/48206215