1005: [HNOI2008]明明的烦恼

Time Limit: 1 Sec  Memory Limit: 162 MB
Submit: 3032  Solved: 1209

Description

自从明明学了树的结构,就对奇怪的树产生了兴趣...... 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树?

Input

第一行为N(0 < N < = 1000),接下来N行,第i+1行给出第i个节点的度数Di,如果对度数不要求,则输入-1

Output

一个整数,表示不同的满足要求的树的个数,无解输出0

Sample Input

3
1
-1
-1

Sample Output

2

HINT

两棵树分别为1-2-3;1-3-2

Source

大意:给出n个点的度数限制,问这些点能组成多少棵树

分析:

首先介绍一种数列Purfer Code:

这种数列是这样子的:对于一棵树(这棵树是不存在根的),每次在数列中加入编号最小的孩子节点(即为度数为1的节点)所对应的父亲节点,并在树中删除此点,不断重复此操作,知道这棵树只剩下两个点

如:

1

/     \

2         3

/     \

4        5

所对应的PurferCode就是:1 2 2

因为第一次删掉3,第二次删掉1,第三次删掉4

剩下2、5两个节点

可以发现,每个PurferCode对应的树是唯一的

我们也可以从PurferCode得到一棵树

首先由PurferCode计算出每点的度数(出现的次数+1),

然后每次(第 i 次)选择度数最少(度数大于1)的点(度数相同时编号较小的优先)与数列第 i 个点连边,然后这两个点的度数都-1,

这样不断操作,会最后的到两个度数为1的点,将这两点相连,即的原来的树

设度数有限制的点的个数为Cnt,度数-1之和为Sum(即为数列中这些点要占的个数),第 i 个点的度数为di

那么可以根据组合数学:

从数列n-2个点中选择Sum个点有C(Sum, n-2)

这Sum个点中组合C(d1-1, Sum)*C(d2-1, Sum-(d1-1))*C(d3-1, Sum-(d1-1)-(d2-1))*........*C(dn - 1, dn - 1)

剩下的n-2-Sum个位置由其余n-Cnt个点组合,有(n-Cnt)^(n-2-Sum)种方案

所以Ans = C(Sum, n-2)*C(d1-1, Sum)*C(d2-1, Sum-(d1-1))*C(d3-1, Sum-(d1-1)-(d2-1))*........*C(dn - 1, dn - 1)*(n-Cnt)^(n-2-Sum)

因为C(Sum, n-2) = (n-2)!/(Sum!*(n-2-Sum)!),

C(d1-1, Sum) = Sum!/((d1-1)!*(Sum-(d1-1))!

C(d2-1, Sum-(d1-1)) = (Sum-(d1-1))!/((d2-1)!*(Sum-(d1-1)-(d2-1))!)

.........

相乘时加粗部分可以相消

最终可得:

Ans =  [ (n-2)!*((n-Cnt)^(n-2-Sum)) ]  /    [(n-2-Sum)!*(d1-1)!*(d2-1)!*.....*(dn-1)!]

写程序时,乘除都会很大,高精度除法和高精度乘高精度都很麻烦,所以可以分解质因数,最终可以只写单精度乘高精度即可。

关于分解x!的质因数

x = p^a   *    k(即x可以整除p^a)

a = x div p  +  x div p^2 + x div p^3 + ..... + x div p^m (p^m为p的阶乘中小于x的最大阶乘,即:  p^m <= x,  p^(m+1) > x)

那么 x!= p1^x1 * p2^x2  *  p3^x3 * ...... * pn^xn

这个是显然的,大家写写就知道

关于分解质因数,可以用O(n)筛素数,这个不讲

综上所述,本题得解

 #include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <deque>
#include <vector>
#include <queue>
#include <iostream>
#include <algorithm>
#include <map>
#include <set>
#include <ctime>
using namespace std;
typedef long long LL;
#define For(i, s, t) for(int i = (s); i <= (t); i++)
#define Ford(i, s, t) for(int i = (s); i >= (t); i--)
#define MIT (2147483647)
#define INF (1000000001)
#define MLL (1000000000000000001LL)
#define sz(x) ((bnt) (x).size())
#define clr(x, y) memset(x, y, sizeof(x))
#define puf push_front
#define pub push_back
#define pof pop_front
#define pob pop_back
#define ft first
#define sd second
#define mk make_pair
inline void SetIO(string Name) {
string Input = Name+".in",
Output = Name+".out";
freopen(Input.c_str(), "r", stdin),
freopen(Output.c_str(), "w", stdout);
} const int N = , LEN = , M = ;
int Prime[N], p;
bool Visit[N];
int n, D[N], Cnt, Sum;
int P1[N], P2[N];
int Ans[LEN], Len; inline void Input() {
scanf("%d", &n);
For(i, , n) scanf("%d", &D[i]);
} inline void EXIT(int Ans) {
printf("%d\n", Ans);
exit();
} inline void BuildPrime() {
clr(Visit, ), p = ;
For(i, , n){
if(!Visit[i]) Prime[++p] = i;
For(j, , p) {
if(i*Prime[j] > n) break;
Visit[i*Prime[j]] = ;
if(i%Prime[j] == ) break;
}
}
} inline void FenJie(int A, int *P) {
For(i, , p) {
int x = Prime[i];
while(x <= A) {
P[i] += A/x;
x *= Prime[i];
}
}
} inline void Mul(int x) {
For(i, , Len) Ans[i] *= x;
For(i, , Len) {
Ans[i+] += Ans[i]/M;
Ans[i] %= M;
}
while(Ans[Len+]) {
Len++;
Ans[Len+] += Ans[Len]/M;
Ans[Len] %= M;
}
} inline void Solve() {
//Check
if(n == ) {
if(D[] == || D[] == -) EXIT();
EXIT();
}
For(i, , n)
if(D[i] == ) EXIT();
Cnt = Sum = ;
For(i, , n)
if(D[i] > ) Cnt++, Sum += D[i]-;
if(Sum > n- || (Sum < n- && n == Cnt)) EXIT(); //Work
BuildPrime();
clr(P1, ), clr(P2, );
FenJie(n-, P1);
FenJie(n--Sum, P2);
For(i, , n)
if(D[i] > ) FenJie(D[i]-, P2);
int m = n-Cnt;
For(i, , p) {
if(m <= ) break;
int t = , k = Prime[i];
while(!(m%k)) m /= k, t++;
P1[i] += t*(n--Sum);
}
For(i, , p) P1[i] -= P2[i]; Ans[] = , Len = ;
For(i, , p)
For(j, , P1[i]) Mul(Prime[i]); printf("%d", Ans[Len]);
Ford(i, Len-, ) printf("%04d", Ans[i]);
cout<<endl;
} int main() {
SetIO("");
Input();
Solve();
return ;
}

下面聊聊用线性筛素数法求欧拉函数

欧拉函数有两个定理

1、当m,n互质时,phi(m*n) = phi(m)*phi(n);

2、phi(p^k) = (p-1)*p^(k-1)

因为1-p^k间只有p的倍数不与其互质,这样的数有p^k/p个,即p^(k-1)个,即phi(p^k) = p^k - p^(k-1) = (p-1)*p^(k-1)

那么当n%p == 0 (显然不互质,p为质数)时,

设n = m * p^k(m与p互质)

phi(n) = phi(m)*phi(p^k) = phi(m)*(p-1)*p^(k-1)

phi(n*p) = phi(m*p^(k+1)) = phi(m)*(p-1)*p^k = phi(n)*p

所以可以得求欧拉函数代码

 for(int i=;i<N;i++)  {
if(!vis[i]) {
prime[++cnt]=i;
phi[i]=i-;
}
for(int k=;k<=cnt&&prime[k]*i<N;k++) {
x=prime[k]*i;
vis[x]=true;
if(i%prime[k]==) {
phi[x]=phi[i]*prime[k];
break;
}
else phi[x]=phi[i]*(prime[k]-);
}
}

bzoj1005 [HNOI2008]明明的烦恼的更多相关文章

  1. bzoj1005: [HNOI2008]明明的烦恼(prufer+高精度)

    1005: [HNOI2008]明明的烦恼 题目:传送门 题解: 毒瘤题啊天~ 其实思考的过程还是比较简单的... 首先当然还是要了解好prufer序列的基本性质啦 那么和1211大体一致,主要还是利 ...

  2. [BZOJ1005] [HNOI2008] 明明的烦恼 (prufer编码)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  3. 【prufer编码+组合数学】BZOJ1005 [HNOI2008]明明的烦恼

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣...... 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Solution 这 ...

  4. BZOJ1005 HNOI2008明明的烦恼(prufer+高精度)

    每个点的度数=prufer序列中的出现次数+1,所以即每次选一些位置放上某个点,答案即一堆组合数相乘.记一下每个因子的贡献分解一下质因数高精度乘起来即可. #include<iostream&g ...

  5. BZOJ1005:[HNOI2008]明明的烦恼(组合数学,Prufer)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  6. [bzoj1005][HNOI2008][明明的烦恼] (高精度+prufer定理)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  7. bzoj1005: [HNOI2008]明明的烦恼 prufer序列

    https://www.lydsy.com/JudgeOnline/problem.php?id=1005 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的 ...

  8. [bzoj1005][HNOI2008]明明的烦恼-Prufer编码+高精度

    Brief Description 给出标号为1到N的点,以及某些点最终的度数,允许在 任意两点间连线,可产生多少棵度数满足要求的树? Algorithm Design 结论题. 首先可以参考这篇文章 ...

  9. [BZOJ1005][HNOI2008]明明的烦恼 数学+prufer序列+高精度

    #include<cstdio> #include<cstring> #include<algorithm> using namespace std; int N; ...

随机推荐

  1. Vector_h

    #ifndef VECTOR_H #define VECTOR_H #include <algorithm> template<typename Object> class V ...

  2. Android错误:Re-installation failed due to different application signatures

    Re-installation failed due to different application signatures (2013-04-20 14:27:32) 转载▼ 标签: 解决方法 问题 ...

  3. java1.8中Lambda表达式reduce聚合测试例子

    public class LambdaTest { public static void main(String[] args) { // 相当于foreach遍历操作结果值 Integer out ...

  4. Swift - 让程序挂起后,能在后台继续运行任务

    1,程序的挂起和退出 由于iOS设备资源有限.当用户点击了home键,或者另一个应用程序启动了.那么原先那个程序便进入后台被挂起,不是退出,只是停止执行代码,同时它的内存被锁定.当应用程序恢复时,它会 ...

  5. 《Linux私房菜》笔记和问题记录

    鸟哥的Linux私房菜简体首页 对Linux的学习侧重于基本命令和运维相关的部分,最后章节的测试问题不错. 1.VIM程序编辑器 1.所有的Linux都会内建VI:很多软件的编辑接口都会主动呼叫VI: ...

  6. C++杂记

    变量就是一个地址,同进程内可以直接访问,要做好线程之间的同步就是了.——摘自CSDN 2015-06-18 16:58:10(注:注意变量的生命周期(作用域就可以不在意))

  7. Shell编程基础教程7--脚本参数的传递

    7.脚本参数的传递    7.1.shift命令        简介:            shift n        每次将参数位置向左偏移n位        例子 #!/bin/bash us ...

  8. Ultra-QuickSort【归并排序典型题目】

    Ultra-QuickSort Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 34470   Accepted: 12382 ...

  9. maven 依赖查询

    该文章源地址:http://xiejianglei163.blog.163.com/blog/static/1247276201362733217604/ 为方便个人使用,转载于此处. http:// ...

  10. ORA-1461 encountered when generating server alert SMG-3500

    Doc ID 461911.1 Patch 6602742 Applies to: Oracle Database - Enterprise Edition - Version 10.2.0.3 an ...