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,反正只是简单环,出不了什么问题。再记录一下结算到但钱违者当前这个环你的切入点是否被选上然后在绕回来时转移给大点就好了。
  

 #include <cstdlib>
#include <cstdio>
#include <cstring>
#include <string>
#include <queue>
#include <algorithm>
#include <cmath>
#include <map>
#define N 1000005
using namespace std;
int n,to[N],zz1,zz,a[N],b[N];
long long va[N],vva[N];
struct ro
{
int to,next,fr;
}road2[N],road[N];
void build2(int x,int y)
{
zz1++;
road2[zz1].to=y;
road2[zz1].next=b[x];
b[x]=zz1;
}
void build(int x,int y,int z)
{
zz++;
road[zz].fr=z;
road[zz].to=y;
road[zz].next=a[x];
a[x]=zz;
}
int dfn[N],low[N],zz2,zz3,top,st[N],bel[N],sta[N],fa[N],size[N];
bool rd[N],rd2[N];
void tar(int x)
{
zz2++;top++;
st[top]=x;
dfn[x]=low[x]=zz2;
rd[x]=rd2[x]=;
for(int i=b[x];i>;i=road2[i].next)
{
int y=road2[i].to;
if(!rd2[y])
{
tar(y);
low[x]=min(low[x],low[y]);
}
else if(rd[y])
{
low[x]=min(dfn[y],low[x]);
}
}
if(dfn[x]==low[x])
{
zz3++;
int v;
do{
v=st[top];
rd[v]=;
top--;
bel[v]=zz3;
sta[zz3]=v;
size[zz3]++;
vva[zz3]+=va[v];
}while(dfn[v]!=low[v]);
}
}
long long f[N][],f2[N][][],f3[N][];
void dfs2(int x,int fr,int tp)
{
if(x==tp&&fr)
{
f2[x][][]=max(f2[fr][][],f2[fr][][]);
f2[x][][]=f2[fr][][];
f[bel[x]][]=max(f2[x][][],f2[x][][]);
return;
}
if(x==tp)
{
f2[x][][]+=f3[x][];
f2[x][][]+=f3[x][];
f2[x][][]+=va[x];
}
else
{
if(fr==tp)
{
f2[x][][]=f2[x][][]=f2[fr][][],f2[x][][]=f2[fr][][];
f2[x][][]+=f3[x][];f2[x][][]+=va[x];
f2[x][][]+=f3[x][];f2[x][][]+=f3[x][];
}
else
{
f2[x][][]=max(f2[fr][][],f2[fr][][]);
f2[x][][]=max(f2[fr][][],f2[fr][][]);
f2[x][][]=f2[fr][][];
f2[x][][]=f2[fr][][];
f2[x][][]+=va[x]+f3[x][];
f2[x][][]+=va[x]+f3[x][];
f2[x][][]+=f3[x][];
f2[x][][]+=f3[x][];
}
}
dfs2(to[x],x,tp);
}
void dfs1(int x)
{
for(int i=a[x];i>;i=road[i].next)
{
int y=road[i].to;
dfs1(y);
if(size[x]==)
{
f[x][]+=max(f[y][],f[y][]);
f[x][]+=f[y][];
}
else
{
f3[road[i].fr][]+=max(f[y][],f[y][]);
f3[road[i].fr][]+=f[y][];
}
}
if(size[x]==)f[x][]+=vva[x];
else
{
dfs2(sta[x],,sta[x]);
}
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%lld%d",&va[i],&to[i]);
build2(to[i],i);
}
for(int i=;i<=n;i++)
{
if(!rd2[i])tar(i);
}
for(int i=;i<=n;i++)
{
for(int j=b[i];j>;j=road2[j].next)
{
int y=road2[j].to;
if(bel[i]!=bel[y])
{
build(bel[i],bel[y],i);
}
}
}
for(int i=;i<=zz3;i++)
{
if(size[i]!=)
{
dfs1(i);
}
}
long long ans=;
for(int i=;i<=zz3;i++)
{
if(size[i]!=)
{
ans+=f[i][];
}
}
printf("%lld\n",ans);
return ;
}

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. PySide——Python图形化界面入门教程(一)

    PySide——Python图形化界面入门教程(一) ——基本部件和HelloWorld 翻译自:http://pythoncentral.io/intro-to-pysidepyqt-basic-w ...

  2. 同城快递公司Postmates近日完成1亿美元融资,美国外卖餐饮迎来一波融资热潮

    美国外卖市场尚未出现一家独大的巨头,一部分原因是与中国的外卖平台相比,在美国,外卖平台要克服的难题可能更多. 4个月之前才完成3亿美元融资的美国同城快递公司Postmates近日又完成1亿美元融资,估 ...

  3. Win10《芒果TV》更新v3.8.40青春版:优化推送策略、新增缓存清理

    芒果TV暑期重磅活动-青春芒果节拉开帷幕,炫酷的3D视觉大秀.王牌IP互动体验馆.众星云集的青春炙燥夜晚会.神秘的芒果吉祥物发布,Win10版<芒果TV>全平台同步更新青春版v3.8.40 ...

  4. ASP如何实现注册后发送激活邮件?

    <% Sub Sendemail(title,content,email) Set jmail = Server.CreateObject("JMAIL.Message") ...

  5. Android零基础入门第26节:layout_gravity和gravity大不同

    原文:Android零基础入门第26节:layout_gravity和gravity大不同 上一期我们一起学习了LinearLayout线性布局的方向.填充模型和权重,本期来一起学习LinearLay ...

  6. Android零基础入门第35节:Android中基于回调的事件处理

    原文:Android零基础入门第35节:Android中基于回调的事件处理 通过前面两期掌握了Android中基于监听的事件处理的五种形式,那么本期一起来学习Android中基于回调的事件处理. 一. ...

  7. asp.net 验证正则表达式 精心整理

    asp.net 验证正则表达式 整数或者小数:^[0-9]+\.{0,1}[0-9]{0,2}$只能输入数字:"^[0-9]*$".只能输入n位的数字:"^\d{n}$& ...

  8. winform子容器随父容器的变化设置

    在设计winform窗体时,因为会很少去调整窗体的大小,这时子控件就会出很尴尬的情况, 通过查看空间的属性,发现有这样两个属性,dock和anchor.这里主要说anchor,官方 解释没太看懂,我的 ...

  9. UWP 设置控件样式四种方法

    1.隐式方法,通过仅指定 Style 的 TargetType.(设置全部的Button样式) <Page.Resources > <Style TargetType="B ...

  10. 在不开启事件循环的线程中使用QTimer(QThread::run函数自带事件循环,在构造函数里创建线程,是一种很有意思的线程用法) good

    引入 QTimer是Qt自带的定时器类,QTimer运行时是依赖于事件循环的,简单来说,在一个不开启事件循环(未调用exec() )的线程中,QTimer是无法使用的.通过分析Qt源码可发现,调用QT ...