• [BZOJ]1005 [HNOI2008]明明的烦恼


    题意

    一棵树,给定每个点的度数,(-1)为无限制,求满足该度数的树的个数。

    题解

    prufer序列的裸题。
    关于prufer序列,网上有更加详细的介绍,这里就不展开说明了,只介绍跟该题相关的性质。

    所有无根树可以跟prufer序列形成双射。
    一棵无根树,每个点在prufer序列出现的次数为它的度数减一。

    这道题显然只要确定有度数限制的那些点,剩下的点任取即可。
    记每个点的度数限制为(d[i])(cnt)为没有度数限制的点的个数

    [ans = frac{A(n - 2, sum (d[i] - 1))}{Pi A(d[i] - 1)} imes cnt ^ {n - 2 - sum (d[i] - 1)} ]

    这道题需要高精度,但是普通的高精度仍然不够,因为组合数增长是(n!)级别的。
    但是显然答案的长度不会很长。
    于是可以记录指数来先进行乘除运算,最后再统一计算答案。
    质因数分解每个要乘的数即可。

    注意判断无解的情况,即度数减一之和大于序列长度 或者 有点的度数为0但是n不为1。

    (BZOJ的题真的是每道题都有收获)

    #include <iostream>
    #include <algorithm>
    #include <cstdio>
    #include <cmath>
    #include <queue>
    #include <map>
    #define Mid ((l + r) >> 1)
    #define lson (rt << 1)
    #define rson (rt << 1 | 1)
    
    using namespace std;
    
    int read(){
    	char c; int num, f = 1;
    	while(c = getchar(),!isdigit(c)) if(c == '-') f = -1; num = c - '0';
    	while(c = getchar(), isdigit(c)) num = num * 10 + c - '0';
    	return f * num;
    }
    const int N = 5009;
    int n, d[N], cnt, res, pri[N], ans[N], len = 1;
    void update(int x, int f) {
    	for(int i = 2; i * i <= x; i++) while(x % i == 0) {
    		pri[i] += f;
    		x /= i;
    	}
    	if(x > 1) pri[x] += f;
    }
    int A(int n, int m, int f) {
    	for(int i = n; i > n - m; i--)
    		update(i, f);
    }
    void times(int x) {
    	for(int i = 1; i <= len; i++) ans[i] *= x;
    	for(int i = 1; i < len; i++) {
    		ans[i + 1] += ans[i] / 10;
    		ans[i] %= 10;
    	}
    	while(ans[len] >= 10) {
    		ans[len + 1] = ans[len] / 10;
    		ans[len] %= 10;
    		len++;
    	}
    }
    void print() {
    	for(int i = len; i; i--)
    		printf("%d", ans[i]);
    }
    signed main()
    {
    	n = read();
    	for(int i = 1; i <= n; i++) {
    		d[i] = read();
    		if(d[i] == -1) cnt++;
    		else {
    			if(d[i] == 0) {
    				if(n == 1) printf("1
    ");
    				else printf("0
    ");
    				return 0;
    			}
    			d[i]--;
    			res += d[i];
    		}
    	}
    	if(res > n - 2) {
    		printf("0
    ");
    		return 0;
    	}
    	ans[1] = 1;
    	A(n - 2, res, 1);
    	for(int i = 1; i <= n; i++) 
    		if(d[i] != -1) 
    			A(d[i], d[i], -1);
    	for(int i = 2; i <= 1000; i++)
    		for(int j = 1; j <= pri[i]; j++)
    			times(i);
    	for(int i = 1; i <= n - 2 - res; i++)
    		times(cnt);
    	print();
    	return 0;
    }
    
  • 相关阅读:
    JAVA中 ReentrantReadWriteLock读写锁详系教程,包会
    传统企业的精益转型之路
    什么时候可以使用极限编程?
    “懒蚂蚁”效应在产品开发过程中的应用
    Vue.config.js配置 最新可用版本
    如何查找一个为NULL的MYSQL字段
    MYSQL 50 基础题 (转载)
    记录一下第一次写 50行 SQL代码
    jwtUtils顾名思意
    对于我们程序员来说,基本面是什么呢?
  • 原文地址:https://www.cnblogs.com/onglublog/p/14493408.html
Copyright © 2020-2023  润新知