题意:
给定一个长5w静态的序列,询问20w次,每次询问查找一个区间内的元素种类数
染色问题神烦啊,最近刚会做,感觉都可以用统一的方法
首先要算出与一个元素相同的最邻近的上一个元素的位置,这样的话只要求出区间内上一位置小于下限的就是元素种类数
——非常有道理的样子,但是第一次见肯定想不到
于是实现了 颜色种类问题->范围内数的个数 的问题转化
然后就简单了,交给权值主席树就好了
算是第二道主席树练习题吧
深井冰错误:构造权值线段树的时候忘记把0构造进去,MDZZ
1 #include <cstdio> 2 #define mid (l+r)/2 3 struct node{int size,ls,rs;} t[4000000]; 4 int cnt=0,n,m,x,y,a[50001],p[50001],c[1000001],root[50001]; 5 int add(int now,int l,int r,int x) 6 { 7 int po=++cnt; 8 t[po]=(node){t[now].size+1,(l<r && x<=mid)?add(t[now].ls,l,mid,x):t[now].ls,(l<r && x>mid)?add(t[now].rs,mid+1,r,x):t[now].rs}; 9 return po; 10 } 11 int que(int x,int y,int l,int r,int z) 12 { 13 if(l==r) return t[y].size-t[x].size; 14 return (z<=mid)?que(t[x].ls,t[y].ls,l,mid,z):t[t[y].ls].size-t[t[x].ls].size+que(t[x].rs,t[y].rs,mid+1,r,z); 15 } 16 int main() 17 { 18 scanf("%d",&n); 19 for(int i=1;i<=n;i++) 20 scanf("%d",&a[i]),p[i]=c[a[i]],c[a[i]]=i,root[i]=add(root[i-1],0,n,p[i]); 21 scanf("%d",&m); 22 for(int i=1;i<=m;i++) 23 scanf("%d%d",&x,&y),printf("%d ",que(root[x-1],root[y],0,n,x-1)); 24 return 0; 25 }
老板娘你还我代码风格!!!