BZOJ1005明明的烦恼 Prufer + 分解質因數 + 高精度
@[高精度, Prufer, 質因數分解]
Description
自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在
任意两点间连线,可产生多少棵度数满足要求的树?
Input
第一行为\(N(0 < N <= 1000)\),
接下来\(N\)行,第\(i + 1\)行给出第\(i\)个节点的度数\(D_i\),如果对度数不要求,则输入\(- 1\)
Output
一个整数,表示不同的满足要求的树的个数,无解输出\(0\)
Sample Input
3
1
-1
-1
Sample Output
2
HINT
两棵树分别为1-2-3; 1-3-2
Solution
Prufer編碼的簡單應用.
Prufer是無根樹的一種編碼, 可以和無根樹之間形成一一對應關係. 生成Prufer編碼的方法很簡單, 大致過程如下(這都不是重點)
一种生成Prufer序列的方法是迭代删点,直到原图仅剩两个点。对于一棵顶点已经经过编号的树T,顶点的编号为{1,2,...,n},在第i步时,移去所有叶子节点(度为1的顶点)中标号最小的顶点和相连的边,并把与它相邻的点的编号加入Prufer序列中,重复以上步骤直到原图仅剩2个顶点。以右边的树为例子,首先在所有叶子节点中编号最小的点是2,和它相邻的点的编号是3,将3加入序列并删除编号为2的点。接下来删除的点是4,5被加入序列,然后删除5,1,此时原图仅剩两个点,Prufer序列构建完成,为{3,5,1,3}
重點在於: 在Prufer編碼中, $$一個節點出現的次數 = 該點的度 - 1$$
有了這個性質, 這題就可以簡單地轉化為求Prufer編碼的排列組合數量.
【思考】怎么计算呢?就用排列组合的原理。我们先考虑不是-1的点。一共有N-2个格子,设当前的点要求为a1(已经减1了),我们可以放的总数是:C(N-2,a1)。这样占了a1个格子。下一个a2的情况就是C(N-2-a1,a2)。依次类推。
然后是-1的情况,我想了很长时间。
原来是这样想的:设最后还有K个格子,且有SUM个-1。我们用DP思路。f[i][j]表示放到最后K个格子中的第i个格子时,放第j种颜色(注意颜色是不下降的,最后只需乘上阶乘即可)的情况数。但是因为要用高精,DP会很麻烦。
感谢HHD大牛的指导:其实就是SUM^K。为什么?对于每一个格子,我们都可以放SUM种情况~~~~。
【优化】为了不超时,用质数表来优化。
實際上就是用分解質因數的方法來優化, 然後再用高精度計算最終答案.
#include<cstdio>
#include<cctype>
#include<cstring>
using namespace std;
inline int read()
{
	int x = 0, flag = 1;
	char c;
	while(! isdigit(c = getchar()))
		if(c == '-')
			flag *= - 1;
	while(isdigit(c))
		x = x * 10 + c - '0', c = getchar();
	return x * flag;
}
void println(int x)
{
    if(x < 0)
        putchar('-'), x *= - 1;
    if(x == 0)
        putchar('0');
    int ans[10], top = 0;
    while(x)
        ans[top ++] = x % 10, x /= 10;
    for(; top; top --)
        putchar(ans[top - 1] + '0');
    putchar('\n');
}
const int N = 1 << 10;
int top;
int isprime[N], prime[N];
void get_prime()
{
	memset(isprime, - 1, sizeof(isprime));
	top = 0;
	for(int i = 2; i < N; i ++)
		if(isprime[i] == - 1)
		{
			isprime[i] = 1, prime[top ++] = i;
			for(int j = i << 1; j < N; j += i)
				isprime[j] = 0;
		}
}
int ans[N];
void C(int n, int m)
{
	for(int i = n; i > n - m; i --)
	{
		int p = i;
		for(int j = 0; j < top; j ++)
			while(! (p % prime[j]))
				ans[j] ++, p /= prime[j];
	}
	for(int i = m; i; i --)
	{
		int p = i;
		for(int j = 0; j < top; j ++)
			while(! (p % prime[j]))
				ans[j] --, p /= prime[j];
	}
}
struct Giant
{
	int dig[1 << 20];
	int top;
	Giant()
	{
		top = 1;
		memset(dig, 0, sizeof(dig));
		dig[0] = 1;
	}
};
void operator *(Giant &x, int y)
{
	for(int i = 0; i < x.top; i ++)
		x.dig[i] *= y;
	for(int i = 0; i < x.top; i ++)
		if(x.dig[i] > 9)
			x.dig[i + 1] += x.dig[i] / 10, x.dig[i] %= 10;
	while(x.dig[x.top])
		x.dig[x.top + 1] = x.dig[x.top] / 10,
		x.dig[x.top] %= 10,
		x.top ++;
}
void println(Giant &x)
{
	for(int i = x.top; i; i --)
		putchar(x.dig[i - 1] + '0');
	putchar('\n');
}
Giant Ans;
int main()
{
	#ifndef ONLINE_JUDGE
	freopen("BZOJ1005.in", "r", stdin);
	freopen("BZOJ1005.out", "w", stdout);
	#endif
	get_prime();
	int n = read();
	int len = 0, cnt = 0;
	//len記錄prufer序列已經佔用的長度, cnt記錄尚未確定度的點數
	for(int i = 0; i < n; i ++)
	{
		int D = read();
		if((! D) || (D - 1 + len > n - 2))
		{
			println(0);
			return 0;
		}
		if(D == - 1)
			cnt ++;
		if(D > 0)
			C(n - 2 - len, D - 1), len += D - 1;
	}
	if(len < n - 2)
	{
		for(int i = 0; i < top; i ++)
			while(! (cnt % prime[i]))
				ans[i] += n - 2 - len, cnt /= prime[i];
	}
	for(int i = 0; i < top; i ++)
		for(int j = 0; j < ans[i]; j ++)
			Ans * prime[i];
	println(Ans);
}
BZOJ1005明明的烦恼 Prufer + 分解質因數 + 高精度的更多相关文章
- bzoj1211树的计数 x bzoj1005明明的烦恼 题解(Prufer序列)
		1211: [HNOI2004]树的计数 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3432 Solved: 1295[Submit][Stat ... 
- bzoj 1005: [HNOI2008]明明的烦恼 prufer编号&&生成树计数
		1005: [HNOI2008]明明的烦恼 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 2248 Solved: 898[Submit][Statu ... 
- BZOJ 1005 [HNOI2008]明明的烦恼 (Prufer编码 + 组合数学 + 高精度)
		1005: [HNOI2008]明明的烦恼 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 5786 Solved: 2263[Submit][Stat ... 
- bzoj 1005 [HNOI2008] 明明的烦恼 (prufer编码)
		[HNOI2008]明明的烦恼 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 5907 Solved: 2305[Submit][Status][Di ... 
- [BZOJ1005] [HNOI2008] 明明的烦恼 (prufer编码)
		Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ... 
- 【bzoj1005】[HNOI2008]明明的烦恼  Prufer序列+高精度
		题目描述 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? 输入 第一行为N(0 < N < = 1000),接下来N行,第i+1行给出第i ... 
- [bzoj1005][HNOI2008]明明的烦恼-Prufer编码+高精度
		Brief Description 给出标号为1到N的点,以及某些点最终的度数,允许在 任意两点间连线,可产生多少棵度数满足要求的树? Algorithm Design 结论题. 首先可以参考这篇文章 ... 
- bzoj1005: [HNOI2008]明明的烦恼   prufer序列
		https://www.lydsy.com/JudgeOnline/problem.php?id=1005 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的 ... 
- BZOJ.1005.[HNOI2008]明明的烦恼(Prufer 高精 排列组合)
		题目链接 若点数确定那么ans = (n-2)!/[(d1-1)!(d2-1)!...(dn-1)!] 现在把那些不确定的点一起考虑(假设有m个),它们在Prufer序列中总出现数就是left=n-2 ... 
随机推荐
- Applied Nonparametric Statistics-lec9
			Ref:https://onlinecourses.science.psu.edu/stat464/print/book/export/html/12 前面我们考虑的情况是:response是连续的, ... 
- POJ:1185-炮兵阵地(状压dp入门)
			炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组 ... 
- Linux学习-什么是例行性工作排程
			那么 Linux 的例行性工作是如何进行排程的呢?所谓的排程就是将这些工作安排执行的流程之意! 咱们的 Linux 排程就是透过 crontab 与 at 这两个东西! Linux 工作排程的种类: ... 
- POJ 3241 曼哈顿距离最小生成树 Object Clustering
			先上几个资料: 百度文库有详细的分析和证明 cxlove的博客 TopCoder Algorithm Tutorials #include <cstdio> #include <cs ... 
- Http协议中的get和post
			Http中post和get区别,是不是用get的方法用post都能办到? Http定义了与服务器交互的不同方法,最基本的方法有4种,分别是GET,POST,PUT,DELETE.URL全称是资源描述符 ... 
- 64位程序调用32DLL解决方案
			最近做一个.NETCore项目,需要调用以前用VB6写的老程序,原本想重写,但由于其调用了大量32DLL,重写后还需要编译为32位才能运行,于是干脆把老代码整个封装为32DLL,然后准备在64位程序中 ... 
- Jeddict:怎样在window系统下,成功打包Angular
			在Jeddict的应用过程中,发现了一个问题.如果前端视图选择Angular(具体版本,需要根据插件版本确定.此处理解为非Angular JS),那么在自动构建的过程中,会发现,每当在安装NodeJS ... 
- poj1006 中国剩余定理&&中国剩余定理解析
			poj 1006 题的思路不是很难的,可以转化数学式: 现设 num 是下一个相同日子距离开始的天数 p,e,i,d 如题中所设! 那么就可以得到三个式子:( num + d ) % 23 == p: ... 
- 【bzoj3513】[MUTC2013]idiots  FFT
			题目描述 给定n个长度分别为a_i的木棒,问随机选择3个木棒能够拼成三角形的概率. 输入 第一行T(T<=100),表示数据组数. 接下来若干行描述T组数据,每组数据第一行是n,接下来一行有n个 ... 
- 刷题总结——寻宝游戏(bzoj3991 dfs序)
			题目: Description 小B最近正在玩一个寻宝游戏,这个游戏的地图中有N个村庄和N-1条道路,并且任何两个村庄之间有且仅有一条路径可达.游戏开始时,玩家可以任意选择一个村庄,瞬间转移到这个村庄 ... 
