这题需要了解一种数列: Purfer Sequence

我们知道,一棵树可以用括号序列来表示,但是,一棵顶点标号(1~n)的树,还可以用一个叫做 Purfer Sequence 的数列表示

一个含有 n 个节点的 Purfer Sequence 有 n-2 个数,Purfer Sequence 中的每个数是 1~n 中的一个数

一个定理:一个 Purfer Sequence 和一棵树一一对应

先看看怎么由一个树得到 Purfer Sequence

由一棵树得到它的 Purfer Sequence 总共需要 n-2 步,每一步都在当前的树中寻找具有最小标号的叶子节点(度为 1),将与其相连的点的标号设为 Purfer Sequence 的第 i 个元素,并将此叶子节点从树中删除,直到最后得到一个长度为 n-2 的 Purfer Sequence 和一个只有两个节点的树

看看下面的例子:

假设有一颗树有 5 个节点,四条边依次为:(1, 2), (1, 3), (2, 4), (2, 5),如下图所示:

第 1 步,选取具有最小标号的叶子节点 3,将与它相连的点 1 作为第 1 个 Purfer Number,并从树中删掉节点 3:

第 2 步,选取最小标号的叶子节点 1,将与其相连的点 2 作为第 2 个 Purfer Number,并从树中删掉点 1:

第 3 步,选取最小标号的叶子节点 4,将与其相连的点 2 作为第 3 个 Purfer Number,并从树中删掉点 4:

最后,我们得到的 Purfer Sequence 为:1 2 2

不难看出,上面的步骤得到的 Purfer Sequence 具有唯一性,也就是说,一个树,只能得到一个唯一的 Purfer Sequence

接下来看,怎么由一个 Purfer Sequence 得到一个树

由 Purfer Sequence 得到一棵树,先将所有编号为 1 到 n 的点的度赋初值为 1,然后加上它在 Purfer Sequence 中出现的次数,得到每个点的度

先执行 n-2 步,每一步,选取具有最小标号的度为 1 的点 u 与 Purfer Sequence 中的第 i 个数 v 表示的顶点相连,得到树中的一条边,并将 u 和 v 的度减一

最后再把剩下的两个度为 1 的点连边,加入到树中

我们可以根据上面的例子得到的 Purfer Sequence :1 2 2 重新得到一棵树

Purfer Sequence 中共有 3 个数,可以知道,它表示的树中共有 5 个点,按照上面的方法计算他们的度为下表所示:

顶点 1 2 3 4 5
2 3 1 1 1

第 1 次执行,选取最小标号度为 1 的点 3 和 Purfer Sequence 中的第 1 个数 1 连边:

将 1 和 3 的度分别减一:

顶点 1 2 3 4 5
1 3 0 1 1

第 2 次执行,选取最小标号度为 1 的点 1 和 Purfer Sequence 中的第 2 个数 2 连边:

将 1 和 2 的度分别减一:

顶点 1 2 3 4 5
0 2 0 1 1

第 3 次执行,将最小标号度为 1 的点 4 和 Purfer Sequence 第 3 个数 2 连边:

将 2 和 4 的度分别减一:

顶点 1 2 3 4 5
0 1 0 0 1

最后,还剩下两个点 2 和 5 的度为 1,连边:

至此,一个 Purfer Sequence 得到的树画出来了,由上面的步骤可知,Purfer Sequence 和一个树唯一对应

综上,一个 Purfer Sequence 和一棵树一一对应

有了 Purfer Sequence 的知识,这题怎么搞定呢?

先不考虑无解的情况,从 Purfer Sequence 构造树的过程中可知,一个点的度数减一表示它在 Purfer Sequence 中出现了几次,那么:

假设度数有限制的点的数量为 cnt,他们的度数分别为:d[i]

另:

那么,在 Purfer Sequence 中的不同排列的总数为:

而剩下的 n-2-sum 个位置,可以随意的排列剩余的 n-cnt 个点,于是,总的方案数就应该是:

化简之后为:

以上题解转自http://www.cnblogs.com/zhj5chengfeng/p/3278557.html

  1. //By BLADEVIL
  2. var
  3. n :longint;
  4. d :array[..] of int64;
  5. a, b, c :array[..] of int64;
  6.  
  7. procedure init;
  8. var
  9. i :longint;
  10. begin
  11. read(n);
  12. for i:= to n do read(d[i]);
  13. end;
  14.  
  15. function mul(s1,s2:ansistring):ansistring;
  16. var
  17. len1, len2 :int64;
  18. i, j :longint;
  19. s :ansistring;
  20.  
  21. begin
  22. fillchar(a,sizeof(a),);
  23. fillchar(b,sizeof(b),);
  24. fillchar(c,sizeof(c),);
  25. len1:=length(s1);
  26. len2:=length(s2);
  27. for i:= to len1 do a[(len1-i) div +]:=a[(len1-i) div +]*+ord(s1[i])-;
  28. for i:= to len2 do b[(len2-i) div +]:=b[(len2-i) div +]*+ord(s2[i])-;
  29.  
  30. len1:=(len1+) div ;
  31. len2:=(len2+) div ;
  32. for i:= to len1 do
  33. for j:= to len2 do
  34. begin
  35. c[i+j-]:=c[i]+a[i]*b[j];
  36. c[i+j]:=c[i+j-] div ;
  37. c[i+j-]:=c[i+j-] mod ;
  38. end;
  39. mul:='';
  40. inc(len1);
  41. for i:=len1 downto do
  42. begin
  43. str(c[i],s);
  44. if c[i]< then mul:=mul+'';
  45. if c[i]< then mul:=mul+'';
  46. if c[i]< then mul:=mul+'';
  47. if c[i]< then mul:=mul+'';
  48. if c[i]< then mul:=mul+'';
  49. if c[i]< then mul:=mul+'';
  50. if c[i]< then mul:=mul+'';
  51. mul:=mul+s;
  52. end;
  53. while (mul[]='') and (length(mul)>) do delete(mul,,);
  54. end;
  55.  
  56. function divide(s:ansistring;x:int64):ansistring;
  57. var
  58. len :int64;
  59. i :longint;
  60.  
  61. begin
  62. fillchar(a,sizeof(a),);
  63. fillchar(c,sizeof(c),);
  64. len:=length(s);
  65. for i:= to len do a[(len-i) div +]:=a[(len-i) div +]*+ord(s[i])-;
  66. len:=(len+) div ;
  67. for i:=len downto do
  68. begin
  69. c[i]:=c[i]+a[i] div x;
  70. a[i-]:=a[i-]+(a[i] mod x)*;
  71. end;
  72. divide:='';
  73. for i:=len downto do
  74. begin
  75. str(c[i],s);
  76. if c[i]< then divide:=divide+'';
  77. if c[i]< then divide:=divide+'';
  78. if c[i]< then divide:=divide+'';
  79. if c[i]< then divide:=divide+'';
  80. if c[i]< then divide:=divide+'';
  81. if c[i]< then divide:=divide+'';
  82. if c[i]< then divide:=divide+'';
  83. divide:=divide+s;
  84. end;
  85. while (divide[]='') and (length(divide)>) do delete(divide,,);
  86. end;
  87.  
  88. procedure main;
  89. var
  90. sum :int64;
  91. flag :boolean;
  92. cnt :int64;
  93. ans, s :ansistring;
  94. i, j :longint;
  95.  
  96. begin
  97. if n= then
  98. begin
  99. if (d[]=) or (d[]=-) then writeln() else writeln();
  100. exit;
  101. end;
  102. sum:=;
  103. flag:=false;
  104. cnt:=;
  105. for i:= to n do if d[i]<>- then
  106. begin
  107. inc(sum,d[i]-);
  108. inc(cnt);
  109. if (d[i]>n-) or (d[i]=) then flag:=true;
  110. end;
  111.  
  112. if flag then
  113. begin
  114. writeln();
  115. exit;
  116. end;
  117. if sum>n- then
  118. begin
  119. writeln();
  120. exit;
  121. end;
  122. flag:=false;
  123. ans:='';
  124. for i:=n--sum to n- do
  125. begin
  126. str(i,s);
  127. ans:=mul(ans,s);
  128. end;
  129. str(n-cnt,s);
  130. for i:= to n--sum do ans:=mul(ans,s);
  131. for i:= to n do
  132. begin
  133. if d[i]<>- then
  134. for j:= to d[i]- do
  135. begin
  136. ans:=divide(ans,j);
  137. end;
  138. end;
  139. writeln(ans);
  140. end;
  141.  
  142. begin
  143. init;
  144. main;
  145. end.

bzoj 1005 组合数学 Purfer Sequence的更多相关文章

  1. BZOJ 1005 [HNOI2008] 明明的烦恼(组合数学 Purfer Sequence)

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

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

    [HNOI2008]明明的烦恼 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5907  Solved: 2305[Submit][Status][Di ...

  3. Purfer Sequence

    原文地址:http://www.cnblogs.com/zhj5chengfeng/archive/2013/08/23/3278557.html 我们知道,一棵树可以用括号序列来表示,但是,一棵顶点 ...

  4. BZOJ 1005 明明的烦恼

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

  5. BZOJ 1005: [HNOI2008]明明的烦恼 Purfer序列 大数

    1005: [HNOI2008]明明的烦恼 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/ ...

  6. BZOJ 1005: [HNOI2008]明明的烦恼( 组合数学 + 高精度 )

    首先要知道一种prufer数列的东西...一个prufer数列和一颗树对应..然后树上一个点的度数-1是这个点在prufer数列中出现次数..这样就转成一个排列组合的问题了.算个可重集的排列数和组合数 ...

  7. BZOJ 1005 明明的烦恼 (组合数学)

    题解:n为树的节点数,d[ ]为各节点的度数,m为无限制度数的节点数. 则               所以要求在n-2大小的数组中插入tot各序号,共有种插法: 在tot各序号排列中,插第一个节点的 ...

  8. BZOJ 1005 [HNOI2008]明明的烦恼 (Prufer编码 + 组合数学 + 高精度)

    1005: [HNOI2008]明明的烦恼 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5786  Solved: 2263[Submit][Stat ...

  9. BZOJ 1005 [HNOI2008]明明的烦恼 purfer序列,排列组合

    1005: [HNOI2008]明明的烦恼 Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少 ...

随机推荐

  1. 前端技术Jquery与Ajax使用总结

    前端技术Jquery与Ajax使用总结 虽然主要是做的后端,但是由于有些时候也要写写前台的界面,因此也就学习了下Jquery和Ajax的一些知识,虽说此次写的这些对于前端大神来说有些班门弄斧的感觉,但 ...

  2. nmon Analyser分析仪

    nmon Analyser官网: https://www.ibm.com/developerworks/community/wikis/home?lang=en#!/wiki/Power+System ...

  3. 把python脚本打包成win可执行文件

    前几天有个朋友找我写一点小东西,写好后把代码发他帮他搞了半天,结果愣是没听懂,就找到了这个办法. 1.导入pyinstaller包, pip install pyinstaller 2.进入到你需要打 ...

  4. 使用hibernate连接Oracle时的权限问题

    在使用hibernate对象关系映射连接和创建表的时候,会涉及到很多权限问题,有些数据库管理会将权限设的很细,我们可以根据后台日志错误和异常信息作出判断. 比如下图所示这个错误(这是我在给银行投产系统 ...

  5. Mybatis 异常记录(1): Invalid bound statement (not found)

    错误信息: org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): com.pingan.cr ...

  6. LeetCode 全解(bug free 训练)

    1.Two Sum Given an array of integers, return indices of the two numbers such that they add up to a s ...

  7. Faster RCNN代码解析

    1.faster_rcnn_end2end训练 1.1训练入口及配置 def train(): cfg.GPU_ID = 0 cfg_file = "../experiments/cfgs/ ...

  8. 容器基础(五): 实现一个简单容器sdocker

    在前面几部分的基础上, 我们更新一下代码,实现一个简单容器 sdocker. sdocker目录构成 linux: # tree . ├── Makefile ├── cpu-test.c # 由cp ...

  9. 扩展欧几里得 求ax+by == n的非负整数解个数

    求解形如ax+by == n (a,b已知)的方程的非负整数解个数时,需要用到扩展欧几里得定理,先求出最小的x的值,然后通过处理剩下的区间长度即可得到答案. 放出模板: ll gcd(ll a, ll ...

  10. input标签与label标签的“合作关系”

    一直忽略了input和label的关系.一次在做自定义单选框的时候又重新捡起来这对“兄弟”. label的for属性和input的id值一致的话,input和label就会组成一个组.例如: < ...