Prufer序列+组合数学


  嗯哼~给定每个点的度数!求树的种数!那么很自然的就想到是用prufer序列啦~(不知道prufer序列的……自己再找找资料吧,这里就不放了,可以去做一下BZOJ1005明明的烦恼)

  那么我们令每个点的度数v[i]-1,得到每个节点在prufer序中的出现次数!

  现在就是求这个prufer序有多少种了……有两种做法:

    1.多重集排列数:n个元素,每种元素有a[i]个,求全排列的方案数,自己随便yy一下就可以得到$$ans=\frac{n!}{\prod (a[i]!)}$$

     意义就是:n个人全排列的数目是N!,然而对于第一种人,这a[1]个人站的顺序不同也只算一种,所以要除以(a[1]!),以此类推,得到上述表达式。

 /**************************************************************
Problem: 1211
User: ProgrammingApe
Language: C++
Result: Accepted
Time:0 ms
Memory:1288 kb
****************************************************************/ //BZOJ 1211
#include<cmath>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
typedef long long LL;
inline LL getint(){
LL r=,v=; char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if (ch=='-') r=-;
for(; isdigit(ch);ch=getchar()) v=v*-''+ch;
return r*v;
}
const int N=;
/*******************template********************/
LL n,m,a[N],ans,fac[N],prime[N],tot;
LL b[N];
bool vis[N];
void ready(int n){
F(i,,n){
if (!vis[i]) prime[++tot]=i;
F(j,,tot){
if (i*prime[j]>n) break;
vis[i*prime[j]]=;
if (i%prime[j]==) break;
}
}
}
void add(int k,int v){
// printf("add %d %d\n",k,v);
F(i,,k){
int x=i;
F(j,,tot) while(x%prime[j]==){
x/=prime[j];
b[j]+=v;
}
}
}
int main(){
#ifndef ONLINE_JUDGE
freopen("tree.in","r",stdin);
freopen("tree.out","w",stdout);
#endif
ready();
n=getint();
if (n==){
a[]=getint();
if (a[]!=) {puts(""); return ;}
else {puts(""); return ;}
}
F(i,,n){
a[i]=getint()-;
if (a[i]< || a[i]>n-) {puts(""); return ;}
m+=a[i];
}
if (m!=n-){puts(""); return ;}
add(m,);
F(i,,n) add(a[i],-);
ans=; F(i,,tot)
F(j,,b[i]) ans*=prime[i];
printf("%lld\n",ans);
return ;
}

(多重集排列数)

    2.组合数:n个位置,拿出a[1]个放第一种元素,再从剩下的(n-a[1])个里面选出a[2]个……所以有$$ans=\prod_{i=1}^{n} \binom{a[i]}{n-\sum_{j=0}^{i-1}(a[j])} $$

 /**************************************************************
Problem: 1211
User: ProgrammingApe
Language: C++
Result: Accepted
Time:4 ms
Memory:1288 kb
****************************************************************/ //BZOJ 1211
#include<cmath>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
typedef long long LL;
inline LL getint(){
LL r=,v=; char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if (ch=='-') r=-;
for(; isdigit(ch);ch=getchar()) v=v*-''+ch;
return r*v;
}
const int N=;
/*******************template********************/
LL n,m,a[N],ans,fac[N],prime[N],tot;
LL b[N];
bool vis[N];
void ready(int n){
F(i,,n){
if (!vis[i]) prime[++tot]=i;
F(j,,tot){
if (i*prime[j]>n) break;
vis[i*prime[j]]=;
if (i%prime[j]==) break;
}
}
}
void add(int k,int v){
// printf("add %d %d\n",k,v);
F(i,,k){
int x=i;
F(j,,tot) while(x%prime[j]==){
x/=prime[j];
b[j]+=v;
}
}
}
inline void C(int a,int b){ add(a,); add(b,-); add(a-b,-);}
int main(){
#ifndef ONLINE_JUDGE
freopen("tree.in","r",stdin);
freopen("tree.out","w",stdout);
#endif
ready();
n=getint();
if (n==){
a[]=getint();
if (a[]!=) {puts(""); return ;}
else {puts(""); return ;}
}
F(i,,n){
a[i]=getint()-;
if (a[i]< || a[i]>n-) {puts(""); return ;}
m+=a[i];
}
if (m!=n-){puts(""); return ;}
F(i,,n) if (a[i]){
C(m,a[i]);
m-=a[i];
}
ans=;
F(i,,tot)
F(j,,b[i]) ans*=prime[i];
printf("%lld\n",ans);
return ;
}

(组合数,处理阶乘时枚举每个数,进行分解质因数)

 /**************************************************************
Problem: 1211
User: ProgrammingApe
Language: C++
Result: Accepted
Time:0 ms
Memory:1288 kb
****************************************************************/ //BZOJ 1211
#include<cmath>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
using namespace std;
typedef long long LL;
inline LL getint(){
LL r=,v=; char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if (ch=='-') r=-;
for(; isdigit(ch);ch=getchar()) v=v*-''+ch;
return r*v;
}
const int N=;
/*******************template********************/
LL n,m,a[N],ans,fac[N],prime[N],tot;
LL b[N];
bool vis[N];
void ready(int n){
F(i,,n){
if (!vis[i]) prime[++tot]=i;
F(j,,tot){
if (i*prime[j]>n) break;
vis[i*prime[j]]=;
if (i%prime[j]==) break;
}
}
}
void add(int k,int v){
// printf("add %d %d\n",k,v);
F(j,,tot){
int x=k;
while(x){
b[j]+=x/prime[j]*v;
x/=prime[j];
}
}
}
inline void C(int a,int b){ add(a,); add(b,-); add(a-b,-);}
int main(){
#ifndef ONLINE_JUDGE
freopen("tree.in","r",stdin);
freopen("tree.out","w",stdout);
#endif
ready();
n=getint();
if (n==){
a[]=getint();
if (a[]!=) {puts(""); return ;}
else {puts(""); return ;}
}
F(i,,n){
a[i]=getint()-;
if (a[i]< || a[i]>n-) {puts(""); return ;}
m+=a[i];
}
if (m!=n-){puts(""); return ;}
F(i,,n) if (a[i]){
C(m,a[i]);
m-=a[i];
// F(j,1,4) cout <<prime[j]<<' '<<b[j]<<endl;
}
// add(m,1);
// F(i,1,n) add(a[i],-1);
ans=;
// F(j,1,4) cout <<prime[j]<<' '<<b[j]<<endl;
F(i,,tot)
F(j,,b[i]) ans*=prime[i];
printf("%lld\n",ans);
return ;
}

(组合数,处理阶乘时直接处理n,分解质因数)

  嗯在看代码之前还有几句要说:这题有$n\leq 150$,所以直接预处理阶乘是不太现实的……所以在这里我用了分解质因数进行快速乘除(这样做高精乘除快->变成质因子次数的加减,普通高精加减快)

  然后也学到了一种快速对阶乘进行因式分解的姿势>_<happy~

1211: [HNOI2004]树的计数

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 1458  Solved: 469
[Submit][Status][Discuss]

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

Source

[Submit][Status][Discuss]

【BZOJ】【1211】【HNOI2004】树的计数的更多相关文章

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

    知道prufer序列就能写...就是求个可重集的排列...先判掉奇怪的情况, 然后答案是(N-2)!/π(d[i]-1)! -------------------------------------- ...

  2. bzoj 1211: [HNOI2004]树的计数 -- purfer序列

    1211: [HNOI2004]树的计数 Time Limit: 10 Sec  Memory Limit: 162 MB Description 一个有n个结点的树,设它的结点分别为v1, v2, ...

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

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

  4. BZOJ 1211[HNOI2004]树的计数 - prufer数列

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

  5. 【刷题】BZOJ 1211 [HNOI2004]树的计数

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

  6. bzoj 1211: [HNOI2004]树的计数

    prufer的应用.. 详细见这篇博客:https://www.cnblogs.com/dirge/p/5503289.html import java.math.BigInteger; import ...

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

    1211: [HNOI2004]树的计数 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2468  Solved: 868 Description 一 ...

  8. BZOJ1211: [HNOI2004]树的计数

    1211: [HNOI2004]树的计数 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1245  Solved: 383[Submit][Statu ...

  9. prufer BZOJ1211: [HNOI2004]树的计数

    以前做过几题..好久过去全忘了. 看来是要记一下... [prufer] n个点的无根树(点都是标号的,distinct)对应一个 长度n-2的数列 所以 n个点的无根树有n^(n-2)种 树 转 p ...

  10. bzoj1211: [HNOI2004]树的计数(prufer序列+组合数学)

    1211: [HNOI2004]树的计数 题目:传送门 题解: 今天刚学prufer序列,先打几道简单题 首先我们知道prufer序列和一颗无根树是一一对应的,那么对于任意一个节点,假设这个节点的度数 ...

随机推荐

  1. FusionCharts X轴显示方式

    本文摘自ITYPE 网站的wangxiao5530 原文网址为:http://wangxiao5530.iteye.com/blog/1454200 1)Wrap Mode (换行模式) <ch ...

  2. ThinkPHP之中的验证码的小示例

    ThinkPHP之中已经封装好了验证码的调用,但是关于手册,缺失了HTML之中以及.实际操作之中的点击ajax就会刷新验证码ajax代码:现在分享一下:看客老爷们注意啦! 放大招啦!!!三分归元气-- ...

  3. Ubuntu10.10的网络配置

    有一阵子着实对Ubuntu的网络配置很迷惑,耐下心来仔细上网找了找,有点小心得,总结一下. 先说下大概的配置过程,再去细究一些情况. 一.配置大概分三类:通过配置文件配置.通过命令配置.通过图形化的网 ...

  4. 有趣的EditView为空时的抖动效果(用户名和密码)--第三方开源--ClearEditText

    ClearEditText在github上的链接地址是:https://github.com/zhangphil/ClearEditText 用法十分简单,在布局中使用ClearEditText,在J ...

  5. 集合删数 (vijos 1545) 题解

    [问题描述] 一个集合有如下元素:1是集合元素:若P是集合的元素,则2 * P +1,4*P+5也是集合的元素,取出此集合中最小的K个元素,按从小到大的顺序组合成一个多位数,现要求从中删除M个数位上的 ...

  6. angularjs2 学习笔记(三) 服务

    在anglar2中服务是什么? 如果在项目中有一段代码是很多组件都要使用的,那么最好的方式就是把它做成服务. 服务是一种在组件中共享功能的机制,当我们使用标签将多个组件组合在一起时我们需要操作一些数据 ...

  7. DoubanFm之设计模式(一)

    前两版DoubanFm写的太戳,第一版可以忽略,当是熟悉WP手机的一些API.. 第二版用了比较多的依赖注入,熟悉了Messenger,过后越写越大,感觉不对,赶快打住..现在开始好好思考各模块了. ...

  8. 【WPF学习日记】——Window的DataContext绑定ViewModel

    1.全局的ViewModel绑定: a)设定全局的ViewModel(App.xaml中): 1 <Application x:Class="MyTest.App" 2 xm ...

  9. JavaWeb之Servlet:Cookie 和 Session

    会话 现实生活中我们会用手机跟对方对话,拿起手机,拨号,然后对面接听,跟着互相通话,最后会话结束. 这个过程也可以用我们的B/S模式来描述: 打开浏览器—>输入地址->发出请求->服 ...

  10. 文件系统 第八次迭代 VFS相关说明

    麻烦访问evernote链接 http://www.evernote.com/shard/s133/sh/53e5b5ac-1192-4910-8bd5-6886218562af/59516c32a5 ...