bzoj 1211 [HNOI2004]树的计数

[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

题解

prufer编码,水题,不知道哪里wrong了,答案是(n-2)!/∏ (di-1)!

 1 #include<cstring>
 2 #include<cmath>
 3 #include<iostream>
 4 #include<algorithm>
 5 #include<cstdio>
 6
 7 #define N 157
 8 #define ll long long
 9 using namespace std;
10 inline int read()
11 {
12     int x=0,f=1;char ch=getchar();
13     while(ch>‘9‘||ch<‘0‘){if (ch==‘-‘) f=-1;ch=getchar();}
14     while(ch<=‘9‘&&ch>=‘0‘)
15     {
16         x=(x<<3)+(x<<1)+ch-‘0‘;
17         ch=getchar();
18     }
19     return x*f;
20 }
21
22 int n;
23 int a[157];
24 int sz,pri[N],num[N],boo[N];
25
26 void solve_fj()
27 {
28     for (int i=2;i<=n;i++)
29     {
30         if (!boo[i]) pri[++sz]=i;
31         for (int j=1;j<=sz&&pri[j]*i<=n;j++)
32         {
33             int t=pri[j]*i;
34             boo[t]=1;
35             if (i%pri[j]==0) break;
36         }
37     }
38     for (int i=2;i<=n-2;i++)
39     {
40         int x=i;
41         for (int j=1;j<=sz;j++)
42             while(x%pri[j]==0)
43             {
44                 num[j]++;
45                 x/=pri[j];
46             }
47     }
48 }
49 ll ksm(ll a,int b)
50 {
51     ll res=1LL;
52     while(b)
53     {
54         if (b%2) res=res*a;
55         a=a*a;
56         b>>=1;
57     }
58     return res;
59 }
60 void solve()
61 {
62     for (int t=1;t<=n;t++)
63     {
64         if (a[t]<=2) continue;
65         for (int i=2;i<=a[t]-1;i++)
66         {
67             int x=i;
68             for (int j=1;j<=sz;j++)
69                 while(x%pri[j]==0)
70                 {
71                     num[j]--;
72                     x/=pri[j];
73                 }
74         }
75     }
76     ll ans=1;
77     for (int i=1;i<=sz;i++)
78         ans*=ksm(pri[i],num[i]);
79     printf("%lld",ans);
80 }
81 int main()
82 {
83     n=read();for (int i=1;i<=n;i++) a[i]=read();
84     solve_fj();
85
86     int num=0;
87     for (int i=1;i<=n;i++)
88         num+=a[i]-1;
89     if (num!=n-2) printf("0");
90     else solve();
91 }
时间: 2024-10-16 11:19:43

bzoj 1211 [HNOI2004]树的计数的相关文章

BZOJ 1211: [HNOI2004]树的计数( 组合数学 )

知道prufer序列就能写...就是求个可重集的排列...先判掉奇怪的情况, 然后答案是(N-2)!/π(d[i]-1)! --------------------------------------------------------------------------- #include<cstdio> #include<algorithm> #include<cstring> using namespace std; typedef long long ll; c

BZOJ 1211: [HNOI2004]树的计数 purfer序列

1211: [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 输出满足条件的树有多少棵

BZOJ 1211 HNOI2004 树的计数 Prufer序列

题目大意:给定一棵树中所有点的度数,求有多少种可能的树 Prufer序列,具体参考[HNOI2008]明明的烦恼 直接乘会爆long long,所以先把每个数分解质因数,把质因数的次数相加相减,然后再乘起来 注意此题无解需要输出0 当n!=1&&d[i]==0时 输出0 当Σ(d[i]-1)!=n-2时输出0 写代码各种脑残--居然直接算了n-2没用阶乘-- #include<cstdio> #include<cstring> #include<iostrea

【BZOJ 1211】 1211: [HNOI2004]树的计数 (prufer序列、计数)

1211: [HNOI2004]树的计数 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2468  Solved: 868 Description 一个有n个结点的树,设它的结点分别为v1, v2, -, vn,已知第i个结点vi的度数为di,问满足这样的条件的不同的树有多少棵.给定n,d1, d2, -, dn,编程需要输出满足d(vi)=di的树的个数. Input 第一行是一个正整数n,表示树有n个结点.第二行有n个数,第i个数表示di,即

【组合数学】【prufer数列】【HNOI 2004】【bzoj 1211】树的计数

1211: [HNOI2004]树的计数 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 1565 Solved: 512 Description 一个有n个结点的树,设它的结点分别为v1, v2, -, vn,已知第i个结点vi的度数为di,问满足这样的条件的不同的树有多少棵.给定n,d1, d2, -, dn,编程需要输出满足d(vi)=di的树的个数. Input 第一行是一个正整数n,表示树有n个结点.第二行有n个数,第i个数表示di,即树

【BZOJ 1211】 [HNOI2004]树的计数

[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] prufer数列的应用 http://www.cnblogs.com/AWCXV/p/7626625.html 这一题没有节点的度数不定. 因此. 所有节点的度数-1的和结果一定要是n-2. 否则就无解. 然后把tot代成n-2就好了. 做法就一样了. (大体思路就是,从n-2个空格里面选出d[i]个空格放节点i,从n-2-d[i]个空格里面选出d[i+1]个空格放节点i+1 (化简一下就成为上文中的式子了. (d[i]==0的

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

[BZOJ1005][HNOI2008]明明的烦恼 Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N(0 < N < = 1000),接下来N行,第i+1行给出第i个节点的度数Di,如果对度数不要求,则输入-1 Output 一个整数,表示不同的满足要求的树的个数,无解输出0 Sample Input 3 1 -1 -1 Sample Outp

【bzoj1211】 HNOI2004—树的计数

http://www.lydsy.com/JudgeOnline/problem.php?id=1211 (题目链接) 题意:一个有n个结点的树,设它的结点分别为v1, v2, …, vn,已知第i个结点vi的度数为di,问满足这样的条件的不同的树有多少棵.给定n,d1, d2, …, dn,编程需要输出满足d(vi)=di的树的个数. Solution  prufer序列,明明的烦恼简化版. 代码: // bzoj1211 #include<algorithm> #include<io

[BZOJ1211][HNOI2004]树的计数(Prufer序列)

题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1211 分析: 关于无根树的组合数学问题肯定想到Prufer序列,类似bzoj1005那题 说下prufer序列的性质: 1.一个无根树对应一个prufer序列 2.一个n个节点无根树对应的prufer序列长度为n-2 3.prufer序列中某节点出现的次数==这个节点在对应的无根树中度数-1 所以这题求无根树的数量等价于求prufer序列的数量. 注意无解的情况就行了.