zoukankan      html  css  js  c++  java
  • CF208E Blood Cousins【线段树合并】

    传送门
    之前用 dsu on tree 搞过一次,这次试着用线段树合并来做,尝试了一下内存回收,基本可以把空间压到 (O(n))。不错不错。
    把询问用向前星来存,速度和空间都比用 vector 存好一些。
    这道题的做法也是先把询问离线,然后每个点建以深度为权值的线段树,先 dfs,然后回溯的时候合并子树的线段树,再查每个点的问题。

    #include <bits/stdc++.h>
    using namespace std;
    const int N=1e5+10;
    int n,dep[N],m,fa[N][22],ans[N],rt[N];
    int head[N],to[N*2],nxt[N*2],tot;
    void add(int u,int v) {to[++tot]=v;nxt[tot]=head[u];head[u]=tot;}
    int qhead[N],qto[N*2],qnxt[N*2],qid[N*2],qtot;
    void qadd(int u,int v,int id) {qto[++qtot]=v;qid[qtot]=id;qnxt[qtot]=qhead[u];qhead[u]=qtot;}
    int read(){
    	int x=0,f=1;char c=getchar();
    	while(c<'0'||c>'9') {if(c=='-')f=-1;c=getchar();}
    	while(c>='0'&&c<='9') {x=x*10+c-'0';c=getchar();}
    	return x*f;
    }
    struct SegTrees{
    	#define mid (l+r>>1)
    	queue<int> que;
    	int sum[N*20],ls[N*20],rs[N*20],tot;
    	int newnode(){if(que.empty()) return ++tot;int u=que.front();que.pop();return u;}
    	void delnode(int id){que.push(id);sum[id]=ls[id]=rs[id]=0;}
    	void upd(int &id,int l,int r,int pos){
    		if(!id) id=newnode();
    		if(l==r) {sum[id]++;return;}
    		if(pos<=mid) upd(ls[id],l,mid,pos);
    		else upd(rs[id],mid+1,r,pos);
    		sum[id]=sum[ls[id]]+sum[rs[id]];
    	}
    	void merge(int &x,int y,int l,int r){
    		if(!x||!y) {x=x+y;return;}
    		if(l==r) {sum[x]+=sum[y];delnode(y);return;}
    		merge(ls[x],ls[y],l,mid);
    		merge(rs[x],rs[y],mid+1,r);
    		sum[x]=sum[ls[x]]+sum[rs[x]];
    		delnode(y);
    	}
    	int ask(int id,int l,int r,int pos){
    		if(!id) return 0;
    		if(l==r) return sum[id];
    		if(pos<=mid) return ask(ls[id],l,mid,pos);
    		else return ask(rs[id],mid+1,r,pos);
    	}
    	#undef mid
    }trs;
    
    void predfs(int u,int rt){
    	dep[u]=dep[rt]+1;fa[u][0]=rt;
    	for(int i=head[u];i;i=nxt[i]) if(to[i]!=rt) predfs(to[i],u);
    }
    
    void dfs(int u,int fa){
    	trs.upd(rt[u],1,n,dep[u]);
    	for(int i=head[u];i;i=nxt[i]){
    		if(to[i]==fa) continue;
    		dfs(to[i],u);
    		trs.merge(rt[u],rt[to[i]],1,n);
    	}
    	for(int i=qhead[u];i;i=qnxt[i])
    		ans[qid[i]]=trs.ask(rt[u],1,n,qto[i]);
    }
    
    int main(){
    	n=read();
    	for(int i=1;i<=n;i++){
    		int u=read();
    		add(u,i);add(i,u);
    	}
    	for(int i=head[0];i;i=nxt[i]) predfs(to[i],0);
    	for(int i=1;i<=20;i++) for(int j=1;j<=n;j++) fa[j][i]=fa[fa[j][i-1]][i-1];
    	m=read();
    	for(int i=1;i<=m;i++){
    		int u=read(),k=read(),rt=u,temp=k;
    		for(int j=20;j>=0;j--) if(k>=(1<<j)) rt=fa[rt][j],k-=(1<<j);
    		if(rt==0) continue;
    		qadd(rt,temp+dep[rt],i);
    	}
    	for(int i=head[0];i;i=nxt[i]) dfs(to[i],0);
    	for(int i=1;i<=m;i++) printf("%d ",ans[i]?ans[i]-1:0);
    	return 0;
    }
    
  • 相关阅读:
    【转】Cocos2d
    unity3d中控制物体移动方法有那些及区别
    Memcached存Session数据、访问安全性、使用场景总结
    [转载]大家都很忙的,请学会帮对方节省时间
    [转载]大家都很忙的,请学会帮对方节省时间
    陆琪:为什么说爱情中“莫欺少年穷”?《秒懂男人》书摘
    SQL Server2012完全备份、差异备份、事务日志备份和还原操作
    SQL Server2012完全备份、差异备份、事务日志备份和还原操作
    SQLSERVER 完整还原 一直显示正在还原解决方法
    SQLSERVER 完整还原 一直显示正在还原解决方法
  • 原文地址:https://www.cnblogs.com/BakaCirno/p/12673617.html
Copyright © 2011-2022 走看看