参考博客https://www.cnblogs.com/dirge/p/5503289.html

(1)prufer数列是一种无根树的编码表示,类似于hash。

一棵n个节点带编号的无根树,对应唯一串长度为n-2的prufer编码。所以一个n阶完全图的生成树个数就是

首先定义无根树中度数为1的节点是叶子节点。

找到编号最小的叶子并删除,序列中添加与之相连的节点编号,重复执行直到只剩下2个节点。

(2)prufer序列转化为无根树。

我们设点集为{1,2...n}。然后我们每次找到点集中没有出现在prufer序列中的最小的点(这一定是这个时刻删除的叶子节点),然后再取出prufer序列中的第一个元素,两个点建边,在将两个点在分别删除。

重要性质:prufer序列中某个编号出现的次数就是这个编号节点的度数-1。

很多时候,无根树树的问题都可以转化为求解prufer序列的问题,从而大大简化了问题。

下面提供了三道例题。

例题:

BSOJ 2503 -- 【HNOI2004】树的计数

Description

  一个有n个结点的树,设它的结点分别为v1, v2, …, vn,已知第i个结点vi的度数为di,问满足这样的条件的不同的树有多少棵。 

  给定n,d1, d2, …, dn,编程需要输出满足d(vi)=di的树的个数。

Input

  第一行是一个正整数n,表示树有n个结点。第二行有n个数,第i个数表示di,即树的第i个结点的度数。其中1<=n<=150,输入数据保证满足条件的树不超过10^17个。

Output

  输出满足条件的树有多少棵。

Sample Input

4 2 1 2 1

Sample Output

2

Hint

  

根据“prufer序列中某个编号出现的次数就是这个编号节点的度数-1”这一性质,这道题就变成了给定元素个数的排列问题。

代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<set>
#include<map>
#include<vector>
#include<ctime>
#define ll long long using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} int n;
ll fac[155];
ll ksm(ll t,ll x) {
ll ans=1;
for(;x;x>>=1,t=t*t)
if(x&1) ans*=t;
return ans;
}
bool vis[150];
ll pri[150],cnt[150];
void pre(int n) {
for(int i=2;i<=n;i++) {
if(!vis[i]) pri[++pri[0]]=i;
for(int j=1;j<=pri[0]&&i*pri[j]<=n;j++) {
vis[i*pri[j]]=1;
if(i%pri[j]==0) break;
}
}
}
void work(ll n,int flag) {
for(int i=1;i<=pri[0];i++) {
for(int j=pri[i];j<=n;j*=pri[i]) {
cnt[i]+=flag*n/j;
}
}
}
int tot;
ll ans[5000];
void Cheng(int t) {
for(int i=1;i<=ans[0];i++) ans[i]*=t;
for(int i=1;i<=ans[0];i++) {
ans[i+1]+=ans[i]/10;
ans[i]%=10;
}
while(ans[ans[0]+1]) {
ans[0]++;
ans[ans[0]+1]+=ans[ans[0]]/10;
ans[ans[0]]%=10;
}
} int main() {
n=Get();
pre(n);
work(n-2,1);
int a;
for(int i=1;i<=n;i++) {
a=Get();
if(!a&&n>1) {cout<<0;return 0;}
if(a>1) work(a-1,-1);
tot+=a;
}
if(tot!=(n-1)*2) return cout<<0,0;
if(n==1) {
if(tot) cout<<0;
else cout<<1;
return 0;
}
if(n==2) return cout<<1,0;
ans[0]=1;
ans[1]=1;
for(int i=1;i<=pri[0];i++)
for(int j=1;j<=cnt[i];j++) Cheng(pri[i]);
for(int i=ans[0];i>=1;i--) cout<<ans[i];
return 0;
}

BSOJ 5553 -- 【模拟试题】wangyurzee的树

Description

wangyurzee有n个各不相同的节点,编号从1到n。wangyurzee想在它们之间连n-1条边,从而使它们成为一棵树。 

可是wangyurzee发现方案数太多了,于是他又给出了m个限制条件,其中第i个限制条件限制了编号为u[i]的节点的度数不能为d[i]。 一个节点的度数,就是指和该节点相关联的边的条数。 这样一来,方案数就减少了,问题也就变得容易了,现在请你告诉wangyurzee连边的方案总数为多少。 答案请对1000000007取模。

Input

第一行输入2个整数n(1<=n<=1000000),m(0<=m<=17)分别表示节点个数以及限制个数。 

第2行到第m+1行描述m个限制条件,第i+1行为2个整数u[i],d[i],表示编号为u[i]的节点度数不能为d[i]。 

为了方便起见,保证1<=ui<=m。同时保证1<=ui<=n,1<=di<=n-1,保证不会有两条完全相同的限制。

Output

输出一行一个整数表示答案。

Sample Input

3 1 1 2

Sample Output

2

显然要容斥。我们就用{总的方案数}-{不满足一个条件的方案数}+{不满足两个条件的方案数}-...

注意有个坑点,就是如果我们枚举的两个条件的u相同,那么方案数直接为0。

然后就是如何计数的问题。假设我们已经定了k个点的度数d[1],d[2]...d[k]。设。那么方案数就是

代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<set>
#include<map>
#include<vector>
#include<ctime>
#define ll long long
#define N 1000005
#define mod 1000000007ll using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} int n,m;
int u[20],d[20];
vector<int>st[N];
ll fac[N],ans;
ll ksm(ll t,ll x) {
ll ans=1;
for(;x;x>>=1,t=t*t%mod)
if(x&1) ans=ans*t%mod;
return ans;
}
ll C(int n,int m) {return fac[n]*ksm(fac[m],mod-2)%mod*ksm(fac[n-m],mod-2)%mod;}
bool vis[N];
void dfs(int v,int flag,int n,int res,ll tot) {
if(v>m) {
(ans+=flag*tot*ksm(res,n)%mod+mod)%=mod;
return ;
}
dfs(v+1,flag,n,res,tot);
if(n>=d[v]&&!vis[u[v]]) {
vis[u[v]]=1;
dfs(v+1,-flag,n-d[v],res-1,tot*C(n,d[v])%mod);
vis[u[v]]=0;
}
}
int main() {
n=Get(),m=Get();
int a;
for(int i=1;i<=m;i++) {
u[i]=Get();d[i]=Get()-1;
}
fac[0]=1;
for(int i=1;i<=n;i++) fac[i]=fac[i-1]*i%mod;
dfs(1,1,n-2,n,1);
cout<<ans;
return 0;
}

BSOJ 5445 -- 【2018雅礼】树

Description

  有n个点,第i个点的限制为度数不能超过ai。

  现在对于每一个s(1<=s<=n),问从这n个点中选出s个点组成有标号无根树的方案数。

Input

  第一行一个整数表示n。

  第二行n个整数a1~an。

Output

  输出仅一行n个整数,第i个整数表示s=i时的答案。

Sample Input

3 2 2 1

Sample Output

3 3 2

Hint

【数据范围】

  对于20%的数据,n≤6。

  对于60%的数据,n≤50。

  对于100%的数据,n≤100。

我们说过,处理无根树计数的问题可以转化为prufer序列的计数问题。

我们设长度为i的prufer序列,用了j个点的方案数。考虑新增一个点v,我们假设它的度数为,然后就可以得到转移方程

最后特判s=1时,答案是n,s>=2是,答案是

代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<set>
#include<map>
#include<vector>
#include<ctime>
#define ll long long
#define mod 1004535809ll
#define N 105 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} int n;
ll f[N][N],g[N][N],c[N][N],w[N];
int main() {
n=Get();
c[0][0]=1;
for(int i=1;i<=n;i++)
for(int j=0;j<=i;j++)
c[i][j]=(!j||i==j)?1:(c[i-1][j-1]+c[i-1][j])%mod;
f[0][0]=1;
for(int i=1;i<=n;i++) {
w[i]=Get();
memcpy(g,f,sizeof(f));
for(int j=0;j<n;j++) {
for(int k=0;k<n;k++) {
if(!f[j][k]) continue ;
for(int q=0;q<w[i]&&q+j<=n-2;q++) {
(g[q+j][k+1]+=f[j][k]*c[q+j][q]%mod)%=mod;
}
}
}
memcpy(f,g,sizeof(f));
}
cout<<n<<" ";
for(int i=2;i<=n;i++) cout<<f[i-2][i]<<" ";
return 0;
}

无根树的计数——prufer序列的更多相关文章

  1. 树的计数 + prufer序列与Cayley公式 学习笔记

    首先是 Martrix67 的博文:http://www.matrix67.com/blog/archives/682 然后是morejarphone同学的博文:http://blog.csdn.ne ...

  2. 树的计数 Prufer序列+Cayley公式

    先安利一发.让我秒懂.. 第一次讲这个是在寒假...然而当时秦神太巨了导致我这个蒟蒻自闭+颓废...早就忘了这个东西了... 结果今天老师留的题中有两道这种的:Luogu P4981 P4430 然后 ...

  3. 树的计数 + prufer序列与Cayley公式(转载)

    原文出处:https://www.cnblogs.com/dirge/p/5503289.html 树的计数 + prufer序列与Cayley公式 学习笔记(转载) 首先是 Martrix67 的博 ...

  4. 【XSY1295】calc n个点n条边无向连通图计数 prufer序列

    题目大意 求\(n\)个点\(n\)条边的无向连通图的个数 \(n\leq 5000\) 题解 显然是一个环上有很多外向树. 首先有一个东西:\(n\)个点选\(k\)个点作为树的根的生成森林个数为: ...

  5. Luogu P2290 [HNOI2004]树的计数 Prufer序列+组合数

    最近碰了$prufer$ 序列和组合数..于是老师留了一道题:P2624 [HNOI2008]明明的烦恼 qwq要用高精... 于是我们有了弱化版:P2290 [HNOI2004]树的计数(考一样的可 ...

  6. BZOJ 1211 HNOI2004 树的计数 Prufer序列

    题目大意:给定一棵树中全部点的度数,求有多少种可能的树 Prufer序列.详细參考[HNOI2008]明明的烦恼 直接乘会爆long long,所以先把每一个数分解质因数.把质因数的次数相加相减.然后 ...

  7. 【BZOJ1211】【HNOI2004】树的计数 prufer序列

    题目描述 给你\(n\)和\(n\)个点的度数,问你有多少个满足度数要求的生成树. 无解输出\(0\).保证答案不超过\({10}^{17}\). \(n\leq 150\) 题解 考虑prufer序 ...

  8. 【BZOJ1005/1211】[HNOI2008]明明的烦恼/[HNOI2004]树的计数 Prufer序列+高精度

    [BZOJ1005][HNOI2008]明明的烦恼 Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可 ...

  9. bzoj1211: [HNOI2004]树的计数 prufer序列裸题

    一个有n个结点的树,设它的结点分别为v1, v2, …, vn,已知第i个结点vi的度数为di,问满足这样的条件的不同的树有多少棵.给定n,d1, d2, …, dn,编程需要输出满足d(vi)=di ...

随机推荐

  1. [Mysql]——通过例子理解事务的4种隔离级别

    SQL标准定义了4种隔离级别,包括了一些具体规则,用来限定事务内外的哪些改变是可见的,哪些是不可见的. 低级别的隔离级一般支持更高的并发处理,并拥有更低的系统开销. 首先,我们使用 test 数据库, ...

  2. S5PV210 时钟体系分析

    S5PV210 时钟体系 如下面时钟结构图所示,S5PV210 中包含 3 大类时钟域, 分别是主系统时钟域(简称 MSYS).显示相关的时钟域(简称 DSYS). 外围设备的时钟域(简称 PSYS) ...

  3. C# ListBox 自动滚动到底部 方法:

    在ListBox中添加一条记录(ListBox.Items.Add方法)后,滚动条会自动回到顶部.我们可能更希望它自动滚动到底部,简要介绍几种方法. 方法一: this.listBox1.Items. ...

  4. MAC下搭建个人博客

    安装homebrew ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/inst ...

  5. JS如何判断一个对象是否为空、是否有某个属性

    一.js判断一个对象是否为空 方法一: let obj1 = {} let obj2 = {a:1} function empty(obj){ for (let key in obj){ return ...

  6. sql server: left join 重复数据

    ---涂聚文 2017-9-28 SELECT VipExamMailProjectId,VipExamMailStaffID FROM VipExamMailRecord WHERE VipExam ...

  7. react-router与react-router-dom使用时的区别

    1.React-router与React-router-dom的API对比 React-router:提供了router的核心api.如Router.Route.Switch等,但没有提供有关dom操 ...

  8. 洛谷P3346 [ZJOI2015]诸神眷顾的幻想乡(广义后缀自动机)

    题意 题目链接 Sol 广义SAM的板子题. 首先叶子节点不超过20,那么可以直接对每个叶子节点为根的子树插入到广义SAM中. 因为所有合法的答案一定是某个叶子节点为根的树上的一条链,因此这样可以统计 ...

  9. IDEA项目搭建九——MybatisPlus多数据库实现

    一.简介 MybatisPlus中引用多数据库时,传统的配置就失效了,需要单独写配置来实现,下面就说一下具体应该如何操作 二.引入MybatisPlus多数据源配置 还是先看一下我的项目结构,Mode ...

  10. WebGL学习笔记(一)

    作者:朱金灿 来源:http://blog.csdn.net/clever101 (一)WebGL是什么? WebGL是一门在网页上显示三维图形的技术,你可以把它理解为把OpenGL从C/S端搬到了B ...