zoukankan      html  css  js  c++  java
  • [Luogu5384][Cnoi2019] 雪松果树

    传送门

    虽然这题是一道二合一,也不算难,但还是学到了很多东西啊,(k) 级儿子个数的五种求法!!我还是觉得四种比较好(

    (k) 级儿子个数有五种求法,你知道么? ——鲁迅

    首先 (k) 级祖先很好求,离线的话dfs的时候开个栈就好了。长链剖分也可以但我不会,倍增什么的就不用说了。

    树上启发式合并

    就是求一个子树里为某一个深度的点的个数嘛,这个明显可以dsu on tree啊,开个桶记录下各种深度的有几个就好了。

    复杂度:(O(nlogn)),应该不能过0_0

    树状数组

    转化为dfs序,就是一个区间里等于某一个数的个数,二维数点弱化版,离线+树状数组。

    复杂度:(O(nlogn))

    二分

    给每个深度开一个vector,按照dfs序把点塞进去,询问时只要在对应深度的vector里二分出区间左右端点就好了。

    这个做法虽然也是 (O(nlogn)) ,但它是在线的,很妙啊!!

    长链剖分

    这个是模板了吧,用一个简单的DP统计一下就好了

    复杂度 (O(n))

    因为我之前其实不会长链剖分所以就写了下代码……

    #include<bits/stdc++.h>
    #define ll long long
    #define fr(i,x,y) for(int i=(x);i<=(y);i++)
    #define rf(i,x,y) for(int i=(x);i>=(y);i--)
    #define frl(i,x,y) for(int i=(x);i<(y);i++)
    using namespace std;
    const int N=1000003;
    const int M=N<<1;
    int n,q;
    int cnt,head[N],Next[M],v[M];
    vector<int> id[N];
    int qk[N];
    
    void read(int &x){
    	char ch=getchar();x=0;
    	for(;ch<'0'||ch>'9';ch=getchar());
    	for(;ch>='0'&&ch<='9';ch=getchar()) x=(x<<3)+(x<<1)+ch-'0';
    }
    
    void add(int x,int y){
    	Next[++cnt]=head[x];
    	head[x]=cnt;
    	v[cnt]=y;
    }
    
    int st[N],L;
    int d[N],bc[N],ls[N],*f[N],*now=ls;
    //vector<int> qry[N];
    void predfs(int x,int fa,int dep){
    	st[dep]=x;
    	//int bc=0;
    	for(auto tmp:id[x])
    	 if (dep>qk[tmp])
    	  id[st[dep-qk[tmp]]].push_back(tmp);
    	id[x].resize(0);
    	for(int i=head[x];i;i=Next[i]){
    	 	predfs(v[i],x,dep+1);
    	 	if (d[v[i]]>d[bc[x]]) bc[x]=v[i];
    	}
    	d[x]=d[bc[x]]+1;
    }
    
    int ans[N];
    void dfs(int x,int fa){
    	f[x][0]=1;
    	if (bc[x]) f[bc[x]]=f[x]+1,dfs(bc[x],x);
    	for(int i=head[x];i;i=Next[i]){
    		int tmp=v[i];
    		if (tmp==bc[x]) continue;
    		f[tmp]=now;now+=d[tmp];
    		dfs(tmp,x);
    		fr(j,1,d[tmp]) f[x][j]+=f[tmp][j-1];
    	}
    	for(auto tmp:id[x])
    	 ans[tmp]=f[x][qk[tmp]]-1;
    }
    
    int main(){
    	read(n);read(q);
    	int x;
    	fr(i,2,n){
    		read(x);
    		add(x,i);
    	}
    	fr(i,1,q){
    		read(x);read(qk[i]);
    		id[x].push_back(i);
    	}
    	predfs(1,0,1);
    	f[1]=now;now+=d[1];
    	fr(i,1,n) for(auto j:id[i]) printf("%d ",j);puts("---");
    	dfs(1,0);
    	fr(i,1,q) printf("%d ",qk[i]==0?0:ans[i]);
    	return 0;
    }
    

    dfs+差分

    这是标算,,想不到……

    前面那个树状数组未免太大材小用了,因为我们只是求区间里等于一个数的个数,并不真的需要树状数组所维护的前缀和。

    我们dfs的时候记录一个 (cnt_i) 表示dfs过的里面深度为 (i) 的有多少个,然后求一个子树里深度为 (d) 的个数只要把dfs这个子树前后的 (cnt_d) 减一减就好了。

  • 相关阅读:
    算法入门7:分支限界法
    算法入门5:贪心算法
    算法入门4:动态规划
    变量
    Java标识符
    Java中的关键字
    Groovy 配置环境变量
    Robot Framework学习笔记(一)------环境搭建
    关于谷歌浏览器(chrome)的一些好用的插件推荐
    关于UML方法学图中类之间的关系:依赖,泛化,关联
  • 原文地址:https://www.cnblogs.com/ymzqwq/p/11809162.html
Copyright © 2011-2022 走看看