[bzoj1005] [洛谷P2624] 明明的烦恼
###Description###
自从明明学了树的结构,就对奇怪的树产生了兴趣…… 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树?
###Input###
第一行为N(0 < N < = 1000),接下来N行,第i+1行给出第i个节点的度数Di,如果对度数不要求,则输入-1
###Output###
一个整数,表示不同的满足要求的树的个数,无解输出0
###Sample Input###
3
1
-1
###Sample Output###
2
###HINT###
两棵树分别为1-2-3;1-3-2
##想法##
这道题用了一个叫做prufer编码的东西(新技能get)
首先prufer编码是啥呢?
引用一下怡红公子的话:
该题运用到了树的prufer编码的性质:
(1)树的prufer编码的实现
不断 删除树中度数为1的最小序号的点,并输出与其相连的节点的序号 直至树中只有两个节点
(2)通过观察我们可以发现
任意一棵n节点的树都可唯一的用长度为n-2的prufer编码表示
度数为m的节点的序号在prufer编码中出现的次数为m-1
(3)怎样将prufer编码还原为一棵树??
从prufer编码的最前端开始扫描节点,设该节点序号为 u ,寻找不在prufer编码的最小序号且没有被标记的节点 v ,连接 u,v,并标记v,将u从prufer编码中删除。扫描下一节点。
这个东西自己画一下挺好理解的。
感觉它最神奇的一点就是这n-2位的编码可以随便填,没有特殊的限制,一个编码就能对应一棵树
一开始我想的是现在n个点中选一个点当根,然后剩余n-1个点都选一个自己的父节点,根据度数判断这个点是多少个点的父节点,但这样有一些问题,比如可能出现环。
而prufer编码就不会出现这样的问题了。
如果我们知道了每个点的度数d[i]
那么满足的树的个数(即prufer编码个数)为 \(\frac{(n-2)!}{\prod(d[i]-1)!}\)
但是题目中有一些点的度数是不知道的。设知道度数的点有m个,它们的 \(\sum d[i]-1\) 为s
那么满足的树的个数为
$
\frac{s!}{\prod(d[i]-1)!} \times C_s \times (n-m) \
=\frac{(n-2)!}{\prod(d[i]-1)! \times (n-2-s)!} \times (n-m)^
$
注意要高精度
##代码##
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 1005;
const int SZ = 1000;
int n;
int d[N];
struct Bint{
int a[N];
int s;
Bint operator * (const int &b) const{
Bint c;
int g=0,f;
for(int i=0;i<s;i++){
f=g+a[i]*b;
c.a[i]=f%SZ;
g=f/SZ;
}
if(g) c.a[s]=g,c.s=s+1;
else c.s=s;
return c;
}
Bint operator / (const int &b) const{
Bint c;
int g=0,f;
for(int i=s-1;i>=0;i--){
f=g*SZ+a[i];
c.a[i]=f/b;
g=f%b;
}
c.s=s;
while(c.s && c.a[c.s-1]==0) c.s--;
return c;
}
Bint operator *= (const int &b) { return *this=*this*b; }
Bint operator /= (const int &b) { return *this=*this/b; }
void output(){ //注意
printf("%d",a[s-1]);
for(int i=s-2;i>=0;i--) {
if(a[i]==0) printf("000");
else if(a[i]<10) printf("00%d",a[i]);
else if(a[i]<100) printf("0%d",a[i]);
else printf("%d",a[i]);
}
printf("\n");
}
}ans;
int main()
{
int s=0,m=0;
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d",&d[i]);
if(d[i]!=-1) m++,s+=d[i]-1;
}
ans.a[0]=1; ans.s=1;
for(int i=n-s-1;i<=n-2;i++) ans*=i;
for(int i=1;i<=n;i++) if(d[i]>1) {
for(int j=2;j<=d[i]-1;j++) ans/=j;
}
for(int i=0;i<n-s-2;i++) ans*=(n-m);
ans.output();
return 0;
}
另一种高精度写法(本质无区别,只是几个月后的写法) 【捂脸】
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
const int N = 1005;
int n,d[N];
struct Bign{
int s[3*N],l;
Bign() { memset(s,0,sizeof(s)); l=0; }
}ans;
Bign mul(Bign a,int x){
Bign c;
for(int i=0,g=0;;i++){
if(i>=a.l && g==0) { c.l=i; break; }
c.s[i]=(a.s[i]*x+g)%10;
g=(a.s[i]*x+g)/10;
}
return c;
}
Bign div(Bign a,int x){
Bign c;
c.l=a.l;
for(int i=a.l-1,g=0;i>=0;i--){
g=g*10+a.s[i];
c.s[i]=g/x;
g=g%x;
}
while(c.l && c.s[c.l-1]==0) c.l--;
return c;
}
void output(Bign a){
for(int i=a.l-1;i>=0;i--)
printf("%d",a.s[i]);
printf("\n");
}
int main()
{
int m,t=0;
scanf("%d",&n);
m=n-2;
for(int i=1;i<=n;i++) {
scanf("%d",&d[i]);
if(d[i]==-1) t++;
else m-=(d[i]-1);
}
ans.s[ans.l++]=1;
for(int i=m+1;i<=n-2;i++) ans=mul(ans,i);
for(int i=1;i<=n;i++)
for(int j=2;j<d[i];j++) ans=div(ans,j);
for(int i=1;i<=m;i++) ans=mul(ans,t);
output(ans);
return 0;
}
[bzoj1005] [洛谷P2624] 明明的烦恼的更多相关文章
- 2021.07.19 P2624 明明的烦恼(prufer序列,为什么杨辉三角我没搞出来?)
2021.07.19 P2624 明明的烦恼(prufer序列,为什么杨辉三角我没搞出来?) [P2624 HNOI2008]明明的烦恼 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn ...
- 【BZOJ1005/1211】[HNOI2008]明明的烦恼/[HNOI2004]树的计数 Prufer序列+高精度
[BZOJ1005][HNOI2008]明明的烦恼 Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可 ...
- 【洛谷P1059 明明的随机数】
题目描述明明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性,他先用计算机生成了N个1到1000之间的随机整数(N≤100),对于其中重复的数字,只保留一个,把其余相同的数去掉,不同的数对应着 ...
- 洛谷 P1059 明明的随机数
题目描述 明明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性,他先用计算机生成了NNN个111到100010001000之间的随机整数(N≤100)(N≤100)(N≤100),对于其中重复 ...
- 洛谷 P1059明明的随机数 & P1068分数线划定 & P1781宇宙总统
题目:https://www.luogu.org/problemnew/show/P1059 思路:STL中的set使用. //#include<bits/stdc++.h> #inclu ...
- 【洛谷2624】[HNOI2008] 明明的烦恼(Python+利用prufer序列结论求解)
点此看题面 大致题意: 给你某些点的度数,其余点度数任意,让你求有多少种符合条件的无根树. \(prufer\)序列 一道弱化版的题目:[洛谷2290][HNOI2004] 树的计数. 这同样也是一道 ...
- 【洛谷2624_BZOJ1005】[HNOI2008] 明明的烦恼(Prufer序列_高精度_组合数学)
题目: 洛谷2624 分析: 本文中所有的 "树" 都是带标号的. 介绍一种把树变成一个序列的工具:Prufer 序列. 对于一棵 \(n\) 个结点的树,每次选出一个叶子(度数为 ...
- 【BZOJ1005】[HNOI2008]明明的烦恼(prufer序列)
[BZOJ1005][HNOI2008]明明的烦恼(prufer序列) 题面 BZOJ 洛谷 题解 戳这里 #include<iostream> #include<cstdio> ...
- 「BZOJ1005」[HNOI2008] 明明的烦恼
「BZOJ1005」[HNOI2008] 明明的烦恼 先放几个prufer序列的结论: Prufer序列是一种对有标号无根树的编码,长度为节点数-2. 具体存在无根树转化为prufer序列和prufe ...
随机推荐
- Linux 内核 NuBus 总线
另一个有趣的, 但是几乎被忘记的, 接口总线是 NuBus. 它被发现于老的 Mac 计算机(那 些有 M68K CPU 家族的). 所有的这个总线是内存映射的(象 M68K 的所有东西), 并且设备 ...
- CodeForce - 1189 D1. Add on a Tree (思维题)
Note that this is the first problem of the two similar problems. You can hack this problem only if y ...
- geoip ip2region2 with spark
上一篇文章中 我使用 maxmind的免费库开发了一个waterdrop的 插件,测试数据发现,国内的有些市级还是不准确,而且香港并不是显示中国,这就不友好了. 找了一下,发下 ip2region 这 ...
- 开发API整理(转)
附送一个 android 源码 查看地址 http://grepcode.com/project/repository.grepcode.com/java/ext/com.google.android ...
- CentOS 7 端口白名单设置
# 查看白名单列表 firewall-cmd --zone=public --list-ports # 添加白名单端口 firewall-cmd --zone=public --add-port=/t ...
- Helm Chart 一键部署 Jenkins
Jenkins Jenkins是一款开源 CI&CD 软件,用于自动化各种任务,包括构建.测试和部署软件.目前提供超过1000个插件来支持构建.部署.自动化, 满足任何项目的需要. Jenki ...
- js对当前时间进行处理
//1.JS获取两个日期之间相差的天数 function getDaysBetween(dateString1, dateString2) { var startDate = Date.parse(d ...
- 小小知识点(十八)U盘中病毒了,System Volume Information文件夹删除不掉
win+R调出命令窗口后搜索cmd,启用cmd命令编辑器,并输入以下命令: attrib "H:\System Volume Information" -s //这句话可以选择 ...
- (一)Django项目架构介绍
项目的架构为: 1.虚拟环境virtualenv 安装Django==2.1.3 安装pymysql 安装mysqlclient 安装其他等 2.项目结构为: 应用APP: blog -- 管理博客 ...
- [Windows10]记一次修复注册表相关血案:该文件没有与之关联的应用来执行该操作。请安装应用,若已经安装应用,请在“默认应用设置”页面中创建关联。
今天闲得蛋疼清理了一下右键菜单,于是在之后某时刻使用Everything的“双击路径列打开目录”功能时发现异常: [Window Title] Everything.exe [Content] 该文件 ...