lyk有一棵树,它想给这棵树重标号。

重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号。
这棵树的烦恼值为所有叶子节点的值的乘积。
lyk想让这棵树的烦恼值最大,你只需输出最大烦恼值对1e9+7取模后的值就可以了。
注意一开始1号节点为根,重标号后这个节点仍然为根。
 
update:数据保证叶子节点个数<=20。
 
例如样例中,将1,2,3,4,5重标号为4,3,1,5,2,此时原来编号为4,5的两个叶子节点的值为3与1,这棵树的烦恼值为3。不存在其它更优解。
Input
  1. 第一行一个数n(1<=n<=100000)。
  2. 接下来n-1行,每行两个数ai,bi(1<=ai,bi<=n),表示存在一条边连接这两个点。
Output
  1. 一行表示答案
Input示例
  1. 5
  2. 1 2
  3. 2 4
  4. 2 3
  5. 3 5
Output示例
3
有一个贪心,就是把小的数尽可能放到靠近叶子节点的点,即从叶子到根逐渐变大
这样小的值就影响尽可能少的叶子
先可以把儿子数为1的点缩掉,这样缩完点后的新树只有40个点左右
我们设$f[S]$表示叶子状态为S的最大烦恼值
为1表示已经分配了值
现在我们要给i分配一个值,转移到S|(1<<i-1)
我们为了满足贪心原则,只有满足子树中叶子全部出现在了S中才有值
否则就会出现矛盾,即从i叶子到根会有比i小的情况
也就是算出这些子树的节点数id,然后叶子i填id+1,计入答案
由于取模不能判大小,所以在维护一个double比大小
  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<algorithm>
  5. #include<cmath>
  6. using namespace std;
  7. typedef long long lol;
  8. struct Node
  9. {
  10. int next,to;
  11. }edge[],edge2[];
  12. int head[],num,num2,head2[],dep[],in[],b[],cnt,tot,n,l[],sz[],s[];
  13. double f[<<];
  14. lol g[<<],Mod=1e9+,w[],sum;
  15. void add(int u,int v)
  16. {
  17. num++;
  18. edge[num].next=head[u];
  19. head[u]=num;
  20. edge[num].to=v;
  21. }
  22. void add2(int u,int v)
  23. {
  24. num2++;
  25. edge2[num2].next=head2[u];
  26. head2[u]=num2;
  27. edge2[num2].to=v;
  28. }
  29. void dfs(int x,int pa,int tp)
  30. {int i;
  31. dep[x]=dep[pa]+;
  32. if (in[x]!=)
  33. {
  34. b[++cnt]=x;w[cnt]=dep[x]-dep[tp];
  35. if (tp!=x)
  36. add2(tp,x);
  37. tp=x;
  38. }
  39. for (i=head[x];i;i=edge[i].next)
  40. {
  41. int v=edge[i].to;
  42. if (v!=pa) dfs(v,x,tp);
  43. }
  44. }
  45. int main()
  46. {int i,u,v,j,k;
  47. cin>>n;
  48. for (i=;i<=n-;i++)
  49. {
  50. scanf("%d%d",&u,&v);
  51. add(u,v);add(v,u);
  52. in[u]++;in[v]++;
  53. }
  54. in[]=-;
  55. dfs(,,);
  56. for (i=;i<=n;i++)
  57. if (in[i]==)
  58. l[++tot]=i,sz[i]=;
  59. for (i=cnt;i>=;i--)
  60. {
  61. for (j=head2[b[i]];j;j=edge2[j].next)
  62. {
  63. int v=edge2[j].to;
  64. sz[b[i]]+=sz[v];
  65. }
  66. }
  67. f[]=g[]=;
  68. for (i=;i<(<<tot)-;i++)
  69. {
  70. for (j=;j<=cnt;j++)
  71. s[b[j]]=;
  72. for (j=;j<=tot;j++)
  73. if (i&(<<j-))
  74. s[l[j]]=;
  75. lol id=;
  76. for (j=cnt;j>=;j--)
  77. {
  78. for (k=head2[b[j]];k;k=edge2[k].next)
  79. {
  80. int v=edge2[k].to;
  81. s[b[j]]+=s[v];
  82. }
  83. if (s[b[j]]==sz[b[j]]) id+=w[j];
  84. }
  85. double as=f[i]*id;
  86. for (j=;j<=tot;j++)
  87. if ((i&(<<j-))==&&as>f[i|(<<j-)])
  88. f[i|(<<j-)]=as,g[i|(<<j-)]=g[i]*id%Mod;
  89. }
  90. cout<<g[(<<tot)-];
  91. }

51nod 1673 树有几多愁的更多相关文章

  1. 51nod 1673 树有几多愁——虚树+状压DP

    题目:http://www.51nod.com/Challenge/Problem.html#!#problemId=1673 建一个虚树. 一种贪心的想法是把较小的值填到叶子上,这样一个小值限制到的 ...

  2. 51nod 1673 树有几多愁(链表维护树形DP+状压DP)

    题意 lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出 ...

  3. [51nod1673]树有几多愁

    lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出最大烦 ...

  4. 刷题总结——树有几多愁(51nod1673 虚树+状压dp+贪心)

    题目: lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输 ...

  5. 51nod1673 树有几多愁 - 贪心策略 + 虚树 + 状压dp

    传送门 题目大意: 给一颗重新编号,叶子节点的值定义为他到根节点编号的最小值,求所有叶子节点值的乘积的最大值. 题目分析: 为什么我觉得这道题最难的是贪心啊..首先要想到 在一条链上,深度大的编号要小 ...

  6. 题解 [51nod1673] 树有几多愁

    题面 解析 这题思路挺秒啊. 本麻瓜终于找了道好题了(还成功把ztlztl大仙拖下水了) 看到叶子节点数<=20就应该是状压啊. 然而DP要怎么写啊? 首先,考虑到编号肯定是从下往上一次增大的, ...

  7. 51nod 1353 树 | 树形DP经典题!

    51nod 1353 树 | 树形DP好题! 题面 切断一棵树的任意条边,这棵树会变成一棵森林. 现要求森林中每棵树的节点个数不小于k,求有多少种切法. 数据范围:\(n \le 2000\). 题解 ...

  8. 51nod 1462 树据结构 | 树链剖分 矩阵乘法

    题目链接 51nod 1462 题目描述 给一颗以1为根的树. 每个点有两个权值:vi, ti,一开始全部是零. Q次操作: 读入o, u, d o = 1 对u到根上所有点的vi += d o = ...

  9. 51nod 1812 树的双直径 题解【树形DP】【贪心】

    老了-稍微麻烦一点的树形DP都想不到了. 题目描述 给定一棵树,边权是整数 \(c_i\) ,找出两条不相交的链(没有公共点),使得链长的乘积最大(链长定义为这条链上所有边的权值之和,如果这条链只有 ...

随机推荐

  1. 记录python接口自动化测试--从excel中读取params参数传入requests请求不生效问题的解决过程(第七目)

    在第六目把主函数写好了,先来运行一下主函数 从截图中可以看到,请求参数打印出来了,和excel中填写的一致 但是每个接口的返回值却都是400,提示参数没有传进去,开始不知道是什么原因(因为excel中 ...

  2. 201621123031 《Java程序设计》第11周学习总结

    作业11-多线程 1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 2. 书面作业 本次PTA作业题集多线程 1. 源代码阅读:多线程程序BounceThread ...

  3. 20145237 《Java程序设计》第七周学习总结

    20145237 <Java程序设计>第七周学习总结 教材学习内容总结 第十三章   一.认识时间与日期   1.时间的度量   在正式认识Java提供了哪些时间处理API之前,得先来了解 ...

  4. NetFPGA-1G-CML Demo --- openflow_switch

    环境 ubuntu 14.04 vivado 15.2 ise 14.6 更多基础配置:http://www.cnblogs.com/wpqwpq/p/6771568.html 运行步骤 step1: ...

  5. HTML5文件操作API

    HTML5文件操作API       一.文件操作API 在之前我们操作本地文件都是使用flash.silverlight或者第三方的activeX插件等技术,由于使用了这些技术后就很难进行跨平台.或 ...

  6. UTF-8 UTF-16 UTF-32 最根本的区别?

    昨天看书的时候突然发现UTF-16 我好像还没见过这个东西  也可能忘记了 反正现在对自己科普一下吧 最根本的区别 UTF-32 把所有的字符都用32bit -- 4个字节 来表示 UTF-16 和 ...

  7. tomcat-theory

    (一) java类:applet,servlet,jsp JSP:.jsp-->.java-->(JVM).classJDK:javac,.java-->.classweb:Serv ...

  8. js判断语句关于true和false后面跟数字或字符串的问题

    我经常在代码中看到很长串判断,看到就头疼,简单的整理一下. 比如:(client.top>=0&&client.left>=0&&client.bottom ...

  9. 新概念英语(1-7)Are you a teacher?

    What is Robert's job? A:I am a new student. My name is Robert. B:Nice to meet you. My name's Sophie. ...

  10. ELK学习总结(2-4)bulk 批量操作-实现多个文档的创建、索引、更新和删除

    bulk 批量操作-实现多个文档的创建.索引.更新和删除 ----------------------------------------------------------------------- ...