Prüfer / Prufer 序列

快速跳转

结论:一个完全图的生成树个数为 n n − 2 n^{n-2} nn2

注意,生成树是指无根树

构造过程

从小到大枚举叶子节点(指度数为1的点),记录其父亲。
在这里插入图片描述

最终为{2,2,3,3,2}

考虑树如何线性建。


void sol1() {
	for(i=1; i<n; ++i) f[i]=read(), ++c[f[i]], ++c[i]; 
	for(i=1; i<=n; ++i) if(c[i]==1) break; 
	p=nw=i; 
	for(i=1; i<=n-2; ++i) {
		ans^=(i*f[nw]); c[f[nw]]--; c[nw]--; 
		if(c[f[nw]]==1 && f[nw]<p) nw=f[nw]; //极度容易错,如果父亲标号小于p,那么就让当前点为父亲
		else {
			while(c[p+1]!=1) ++p; 
			++p; nw=p; 
		}
	}
	printf("%lld", ans); 
}

注意代码中的易错点。

还原回去同理。那一行也很容易错。

void sol2() {
	for(i=1; i<=n-2; ++i) a[i]=read(), ++c[a[i]]; 
	for(i=1; i<=n; ++i) if(c[i]==0) break; 
	p=nw=i; 
	for(i=1; i<=n-2; ++i) {
		f[nw]=a[i]; --c[a[i]]; --c[nw]; 
		if(c[a[i]]==0 && a[i]<p) nw=a[i]; 
		else {
			while(c[p+1]!=0) ++p; 
			++p; nw=p; 
		}
	}
	f[nw]=n; 
	for(i=1; i<=n-1; ++i) ans^=(i*f[i]); 
	printf("%lld", ans); 
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值