1040: [ZJOI2008]骑士

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 5368  Solved: 2044
[Submit][Status][Discuss]

Description

  Z国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英。他们劫富济贫,惩恶扬善,受到社会各
界的赞扬。最近发生了一件可怕的事情,邪恶的Y国发动了一场针对Z国的侵略战争。战火绵延五百里,在和平环境
中安逸了数百年的Z国又怎能抵挡的住Y国的军队。于是人们把所有的希望都寄托在了骑士团的身上,就像期待有一
个真龙天子的降生,带领正义打败邪恶。骑士团是肯定具有打败邪恶势力的能力的,但是骑士们互相之间往往有一
些矛盾。每个骑士都有且仅有一个自己最厌恶的骑士(当然不是他自己),他是绝对不会与自己最厌恶的人一同出
征的。战火绵延,人民生灵涂炭,组织起一个骑士军团加入战斗刻不容缓!国王交给了你一个艰巨的任务,从所有
的骑士中选出一个骑士军团,使得军团内没有矛盾的两人(不存在一个骑士与他最痛恨的人一同被选入骑士军团的
情况),并且,使得这支骑士军团最具有战斗力。为了描述战斗力,我们将骑士按照1至N编号,给每名骑士一个战
斗力的估计,一个军团的战斗力为所有骑士的战斗力总和。

Input

  第一行包含一个正整数N,描述骑士团的人数。接下来N行,每行两个正整数,按顺序描述每一名骑士的战斗力
和他最痛恨的骑士。

Output

  应包含一行,包含一个整数,表示你所选出的骑士军团的战斗力。

Sample Input

3
10 2
20 3
30 1

Sample Output

30

HINT

N ≤ 1 000 000,每名骑士的战斗力都是不大于 1 000 000的正整数。

  在开头先吐槽一下题面,”劫富济贫“是怎么得到社会"各界"的好评的?!  开个玩笑,这道题可以说是Bzoj1214枪战Maf的加强版没做过的可以先去做一下(附题解:[POI2008]枪战Maf题解)。
  对于这道题的种种性质我就不多说了,在那篇题解里基本都分析了。
  这两道题最大的不同就是这道题并不能用贪心做,也就是我们并不能再通过拓扑去得到答案。而是利用他的另一个性质:树。
  我们可以注意到,只要我们缩一下边,那么留给我们的就是一个树,或者树林,那么对于树的叶子节点,他们一定只是由一个骑士组成,所以我们只要简单的用两种状态记录一下选他和不选他的答案就好了。当我们爬到树根时,我们不必急着直接转移给他,而是转移给实际向该叶节点伸边的那个骑士,然后再环中随便找一个点开始一个dfs,反正只是简单环,出不了什么问题。再记录一下结算到但钱违者当前这个环你的切入点是否被选上然后在绕回来时转移给大点就好了。
  

  1. #include <cstdlib>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <string>
  5. #include <queue>
  6. #include <algorithm>
  7. #include <cmath>
  8. #include <map>
  9. #define N 1000005
  10. using namespace std;
  11. int n,to[N],zz1,zz,a[N],b[N];
  12. long long va[N],vva[N];
  13. struct ro
  14. {
  15. int to,next,fr;
  16. }road2[N],road[N];
  17. void build2(int x,int y)
  18. {
  19. zz1++;
  20. road2[zz1].to=y;
  21. road2[zz1].next=b[x];
  22. b[x]=zz1;
  23. }
  24. void build(int x,int y,int z)
  25. {
  26. zz++;
  27. road[zz].fr=z;
  28. road[zz].to=y;
  29. road[zz].next=a[x];
  30. a[x]=zz;
  31. }
  32. int dfn[N],low[N],zz2,zz3,top,st[N],bel[N],sta[N],fa[N],size[N];
  33. bool rd[N],rd2[N];
  34. void tar(int x)
  35. {
  36. zz2++;top++;
  37. st[top]=x;
  38. dfn[x]=low[x]=zz2;
  39. rd[x]=rd2[x]=;
  40. for(int i=b[x];i>;i=road2[i].next)
  41. {
  42. int y=road2[i].to;
  43. if(!rd2[y])
  44. {
  45. tar(y);
  46. low[x]=min(low[x],low[y]);
  47. }
  48. else if(rd[y])
  49. {
  50. low[x]=min(dfn[y],low[x]);
  51. }
  52. }
  53. if(dfn[x]==low[x])
  54. {
  55. zz3++;
  56. int v;
  57. do{
  58. v=st[top];
  59. rd[v]=;
  60. top--;
  61. bel[v]=zz3;
  62. sta[zz3]=v;
  63. size[zz3]++;
  64. vva[zz3]+=va[v];
  65. }while(dfn[v]!=low[v]);
  66. }
  67. }
  68. long long f[N][],f2[N][][],f3[N][];
  69. void dfs2(int x,int fr,int tp)
  70. {
  71. if(x==tp&&fr)
  72. {
  73. f2[x][][]=max(f2[fr][][],f2[fr][][]);
  74. f2[x][][]=f2[fr][][];
  75. f[bel[x]][]=max(f2[x][][],f2[x][][]);
  76. return;
  77. }
  78. if(x==tp)
  79. {
  80. f2[x][][]+=f3[x][];
  81. f2[x][][]+=f3[x][];
  82. f2[x][][]+=va[x];
  83. }
  84. else
  85. {
  86. if(fr==tp)
  87. {
  88. f2[x][][]=f2[x][][]=f2[fr][][],f2[x][][]=f2[fr][][];
  89. f2[x][][]+=f3[x][];f2[x][][]+=va[x];
  90. f2[x][][]+=f3[x][];f2[x][][]+=f3[x][];
  91. }
  92. else
  93. {
  94. f2[x][][]=max(f2[fr][][],f2[fr][][]);
  95. f2[x][][]=max(f2[fr][][],f2[fr][][]);
  96. f2[x][][]=f2[fr][][];
  97. f2[x][][]=f2[fr][][];
  98. f2[x][][]+=va[x]+f3[x][];
  99. f2[x][][]+=va[x]+f3[x][];
  100. f2[x][][]+=f3[x][];
  101. f2[x][][]+=f3[x][];
  102. }
  103. }
  104. dfs2(to[x],x,tp);
  105. }
  106. void dfs1(int x)
  107. {
  108. for(int i=a[x];i>;i=road[i].next)
  109. {
  110. int y=road[i].to;
  111. dfs1(y);
  112. if(size[x]==)
  113. {
  114. f[x][]+=max(f[y][],f[y][]);
  115. f[x][]+=f[y][];
  116. }
  117. else
  118. {
  119. f3[road[i].fr][]+=max(f[y][],f[y][]);
  120. f3[road[i].fr][]+=f[y][];
  121. }
  122. }
  123. if(size[x]==)f[x][]+=vva[x];
  124. else
  125. {
  126. dfs2(sta[x],,sta[x]);
  127. }
  128. }
  129. int main()
  130. {
  131. scanf("%d",&n);
  132. for(int i=;i<=n;i++)
  133. {
  134. scanf("%lld%d",&va[i],&to[i]);
  135. build2(to[i],i);
  136. }
  137. for(int i=;i<=n;i++)
  138. {
  139. if(!rd2[i])tar(i);
  140. }
  141. for(int i=;i<=n;i++)
  142. {
  143. for(int j=b[i];j>;j=road2[j].next)
  144. {
  145. int y=road2[j].to;
  146. if(bel[i]!=bel[y])
  147. {
  148. build(bel[i],bel[y],i);
  149. }
  150. }
  151. }
  152. for(int i=;i<=zz3;i++)
  153. {
  154. if(size[i]!=)
  155. {
  156. dfs1(i);
  157. }
  158. }
  159. long long ans=;
  160. for(int i=;i<=zz3;i++)
  161. {
  162. if(size[i]!=)
  163. {
  164. ans+=f[i][];
  165. }
  166. }
  167. printf("%lld\n",ans);
  168. return ;
  169. }

Bzoj 1040 [ZJOI2008]骑士 题解的更多相关文章

  1. [BZOJ 1040][ZJOI2008]骑士

    1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 5403  Solved: 2060[Submit][Status ...

  2. BZOJ 1040: [ZJOI2008]骑士 基环加外向树

    1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1190  Solved: 465[Submit][Status] ...

  3. bzoj 1040: [ZJOI2008]骑士 環套樹DP

    1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1755  Solved: 690[Submit][Status] ...

  4. bzoj 1040: [ZJOI2008]骑士 树形dp

    题目链接 1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3054  Solved: 1162[Submit][S ...

  5. [BZOJ 1040] [ZJOI2008] 骑士 【基环+外向树DP】

    题目链接:BZOJ - 1040 题目分析 这道题目的模型就是一个图,不一定联通,每个连通块的点数等于边数. 每个连通块都是一个基环+外向树.即树上增加了一条边. 如果是树,就可以直接树形DP了.然而 ...

  6. BZOJ 1040: [ZJOI2008]骑士 | 在基环外向树上DP

    题目: http://www.lydsy.com/JudgeOnline/problem.php?id=1040 题解: 我AC了 是自己写的 超开心 的 考虑断一条边 这样如果根节点不选答案一定正确 ...

  7. bzoj 1040 [ZJOI2008]骑士(基环外向树,树形DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1040 [题意] 给一个基环森林,每个点有一个权值,求一个点集使得点集中的点无边相连且权 ...

  8. BZOJ 1040: [ZJOI2008]骑士 [DP 环套树]

    传送门 题意:环套树的最大权独立集 一开始想处理出外向树树形$DP$然后找到环再做个环形$DP$ 然后看了看别人的题解其实只要断开环做两遍树形$DP$就行了...有道理! 注意不连通 然后洛谷时限再次 ...

  9. BZOJ 1040: [ZJOI2008]骑士(基环树dp)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1040 题意: 思路: 这是基环树,因为每个人只会有一个厌恶的人,所以每个节点只会有一个父亲节点,但是 ...

随机推荐

  1. .NET VS 自定义新建代码文件模板

    参考:http://www.cnblogs.com/fightingtong/p/3765914.html 在VS中新建文件时,可使用模板在文件中生成指定内容.只需要把IDE安装目录下的模板进行修改保 ...

  2. win10 uwp 异步转同步

    原文:win10 uwp 异步转同步 有很多方法都是异步,那么如何从异步转到同步? 可以使用的方法需要获得是否有返回值,返回值是否需要. 如果需要返回值,使用GetResults 如从文件夹获取文件: ...

  3. linux下计划任务学习记录

    0x01 计划任务简介 linux 中计划任务主要分为”循环执行”和”只执行一次”两种,分别对应的时 crond 服务 和 atd 服务: 0x02 只执行一次的计划任务 0x02.1 atd 服务说 ...

  4. WPF 鼠标在图片Image上悬停时切换更改设置图片源Source

    // 无效的写法,图片不会被切换 <Image Margin="0,0,0,0" Width="50" Height="50" Sou ...

  5. 简单实用SQL脚本Part:查找SQL Server 自增ID值不连续记录

    原文:简单实用SQL脚本Part:查找SQL Server 自增ID值不连续记录 在很多的时候,我们会在数据库的表中设置一个字段:ID,这个ID是一个IDENTITY,也就是说这是一个自增ID.当并发 ...

  6. 浏览器禁用cookie后php如何保持session会话-use_trans_sid机制

    原文:浏览器禁用cookie后php如何保持session会话-use_trans_sid机制 为防止浏览器禁用cookie导致服务器会话无法保持,php开发了一个机制,该机制开启后,浏览器发起请求后 ...

  7. Office Add-in Model 为 Outlook Mail Add-in 提供的 JavaScript API 介绍

    本文所讨论的 Mailbox API是指在 Mail Add-in 中可调用的 JavaScript API.开发者可以利用这些API 实现 Add-in 和 Outlook 的交互(数据读取与写入) ...

  8. windows Service 之调试过程(附加到进程里调试,而且启动时间不能超过30秒)

    最近第一次用C#写了一个windows service ,其实实现的内容比较简单.就是启动remoting 连接,但是调试相对初次写windws service 的我来说,比较烦.没有经验,而且没办法 ...

  9. 电商、P2P等大型互联网系统包含哪些业务模块?

    01 前言 在互联网飞速发展的时代,各大互联网公司正在进行激烈的竞争,业务模式也在不断的扩张,这种现状使得目前各大公司的架构系统面临着极大的挑战,而对于我们普通的软件开发者而言,如果你仅仅了解过一些关 ...

  10. Linux/windows com串口 java 接收数据 并解析 web程序

    1.首先应公司要求再 com 口本来使用 .net 由于 .net 适用 linux 太麻烦 改为java 准备工作 准备 RXTXconmm.jar(版本很重要) 因为版本问题我搞了一天. 主要讲述 ...