luoguP2597 [ZJOI2012]灾难

题意

这题思路好奇怪啊

见到有向无环图显然是要拓朴排序,不妨按照被吃向吃连边,那么\(x\)灭绝当且仅当x的入点都灭绝,于是考虑怎样x的入点都灭绝

比如4号节点,它灭绝当且仅当2和3灭绝,2和3灭绝当且仅当1灭绝,我们发现1是2和3的lca

于是得出这样一个做法:将每个点\(x\)和灭绝后能灭绝\(x\)的点\(y\)连边\((y->x)\),y是所有x的父亲的lca,这样显然是棵树,每个点的灾难度即子树大小减一

建树可以边拓朴排序边维护倍增数组,可以见代码

code:

#include<bits/stdc++.h>
using namespace std;
const int maxn=70010;
const int maxm=2000010;
const int inf=0x3f3f3f3f;
int n,cnt1,cnt2,cnt,t;
int head1[maxn],head2[maxn],head[maxn],deg[maxn],dep[maxn],size[maxn];
int f[maxn][20];
struct edge{int to,nxt;}e[maxm],e1[maxm],e2[maxm];
inline void add1(int u,int v){e1[++cnt1].nxt=head1[u];head1[u]=cnt1;e1[cnt1].to=v;}
inline void add2(int u,int v){e2[++cnt2].nxt=head2[u];head2[u]=cnt2;e2[cnt2].to=v;}
inline void	add(int u,int v){e[++cnt].nxt=head[u];head[u]=cnt;e[cnt].to=v;}
inline int lca(int x,int y)
{
	if(dep[x]>dep[y])swap(x,y);
	for(int i=t;~i;i--)if(dep[f[y][i]]>=dep[x])y=f[y][i];
	if(x==y)return x;
	for(int i=t;~i;i--)if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
	return f[x][0];
}
inline void topsort()
{
	queue<int>q;
	for(int i=1;i<=n;i++)if(!deg[i])add1(n+1,i),add2(i,n+1),deg[i]++;
	q.push(n+1);dep[n+1]=1;
	while(!q.empty())
	{
		int x=q.front();q.pop();
		for(int i=head1[x];i;i=e1[i].nxt)
		{
			int y=e1[i].to;deg[y]--;
			if(!deg[y])
			{
				int z=0;
				for(int j=head2[y];j;j=e2[j].nxt)
					if(!z)z=e2[j].to;
					else z=lca(z,e2[j].to);
				add(z,y);f[y][0]=z;dep[y]=dep[z]+1;
				for(int j=1;j<=t;j++)f[y][j]=f[f[y][j-1]][j-1];
				q.push(y);
			}
		}
	}
}
void dfs(int x)
{
	size[x]=1;
	for(int i=head[x];i;i=e[i].nxt)
	{
		int y=e[i].to;
		dfs(y);size[x]+=size[y];
	}
}
int main()
{
	scanf("%d",&n);t=(int)log2(n)+1;
	for(int i=1;i<=n;i++)
	{
		int x;
		while(~scanf("%d",&x)&&x)add1(x,i),add2(i,x),deg[i]++;
	}
	topsort();
	/*for(int x=1;x<=n+1;x++)
	{
		printf("now::%d\n",x);
		for(int i=head[x];i;i=e[i].nxt)printf("%d ",e[i].to);
		puts("");
	}*/
	dfs(n+1);
	for(int i=1;i<=n;i++)printf("%d\n",size[i]-1);
	return 0;
}
posted @ 2019-12-03 16:01  nofind  阅读(97)  评论(0编辑  收藏  举报