【基环树/树形DP】BZOJ1040-[ZJOI2008]骑士

【题目大意】

有n个骑士,给出他们的能力值和最痛恨的一位骑士。选出一个骑士军团,使得军团内没有矛盾的两人(不存在一个骑士与他最痛恨的人一同被选入骑士军团的情况),并且,使得这支骑士军团最具有战斗力,求战斗力的最大值。

【思路】

首先yy一下,可以知道这是一个基环森林。我们可以用以下方法:

首先在每一棵基环树的环上任意找到一条边(用dfs来实现),记它的两个端点为u和v。然后删掉这条边(我这里用的方法是记录u,v在对方容器中的位置,并在后续操作中忽略这条边)。由于u和v不能同时取,在删掉u和v之间的边后存在以下两种情况:

令g[i]为不取i时i及其子树的最大战斗力总和,f[i]则表示取i。

(1)u不取,v任意。则以u为根进行树形DP,结果为g[u];

(2)v不取,u任意。则以v为根进行树形DP,结果为g[v]。

然后ans+max(g[u],g[v])。

树形DP的过程非常地常规,就不赘述了 。

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<algorithm>
  5 #include<vector>
  6 using namespace std;
  7 const int MAXN=1000000+500;
  8 typedef long long ll;
  9 vector<int> E[MAXN];
 10 int n,power[MAXN],hate[MAXN];
 11 int vis[MAXN];
 12 int U,V;
 13 ll g[MAXN],f[MAXN];
 14
 15 void addedge(int u,int v)
 16 {
 17     E[u].push_back(v);
 18     E[v].push_back(u);
 19 }
 20
 21 void dfs(int u,int fr)
 22 {
 23     vis[u]=1;
 24     for (int i=0;i<E[u].size();i++)
 25     {
 26         int to=E[u][i];
 27         if (to!=fr)
 28         {
 29             if (!vis[to]) dfs(to,u);
 30             else
 31             {
 32                 vis[to]=1;
 33                 U=u;V=to;
 34                 return;
 35             }
 36         }
 37     }
 38 }
 39
 40 void TreeDP(int u,int fr,int rt,int ban)
 41 {
 42     vis[u]=1;
 43     f[u]=power[u];
 44     g[u]=0;
 45     for (int i=0;i<E[u].size();i++)
 46     {
 47         int to=E[u][i];
 48         if (u==rt && i==ban) continue;
 49         if (to!=fr && to!=rt)
 50         {
 51             TreeDP(to,u,rt,ban);
 52             f[u]+=g[to];
 53             g[u]+=max(g[to],f[to]);
 54         }
 55     }
 56 }
 57
 58 void init()
 59 {
 60     scanf("%d",&n);
 61     for (int i=1;i<=n;i++)
 62     {
 63         scanf("%d%d",&power[i],&hate[i]);
 64         addedge(i,hate[i]);
 65     }
 66 }
 67
 68 void get_ans()
 69 {
 70     memset(vis,0,sizeof(vis));
 71     ll ans=0;
 72     for (int i=1;i<=n;i++)
 73         if (!vis[i])
 74         {
 75             dfs(i,-1);
 76             int banu,banv;
 77             for (int i=0;i<E[U].size();i++) if (E[U][i]==V)
 78             {
 79                 banu=i;
 80                 break;
 81             }
 82             for (int i=0;i<E[V].size();i++) if (E[V][i]==U)
 83             {
 84                 banv=i;
 85                 break;
 86             }
 87
 88
 89             TreeDP(U,-1,U,banu);
 90             ll uans=g[U];
 91
 92             TreeDP(V,-1,V,banv);
 93             ll vans=g[V];
 94             ans+=max(uans,vans);
 95         }
 96     cout<<ans<<endl;
 97 }
 98
 99 int main()
100 {
101     init();
102     get_ans();
103     return 0;
104 } 
时间: 2024-10-13 10:48:39

【基环树/树形DP】BZOJ1040-[ZJOI2008]骑士的相关文章

bzoj 1040: [ZJOI2008]骑士【基环树+树形dp】

没考虑可以连着两个不选--直接染色了 实际上是基环森林,对于每棵基环树,dfs找出一个环边,然后断掉这条边,分别对这条边的两端点做一边treedp,取max加进答案里 treedp是设f[u]为选u点,g[u]为不选u点,然后随便转移一下就行了 #include<iostream> #include<cstdio> using namespace std; const int N=1000005; int n,h[N],cnt=1,x,y,eg; long long a[N],f[

【BZOJ-2286】消耗战 虚树 + 树形DP

2286: [Sdoi2011消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2120  Solved: 752[Submit][Status][Discuss] Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军胜利在望.已知在其他k个岛屿上有丰富能源,为了防止敌军获取能源,我军的任务是炸毁

【BZOJ-3572】世界树 虚树 + 树形DP

3572: [Hnoi2014]世界树 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1084  Solved: 611[Submit][Status][Discuss] Description 世界树是一棵无比巨大的树,它伸出的枝干构成了整个世界.在这里,生存着各种各样的种族和生灵,他们共同信奉着绝对公正公平的女神艾莉森,在他们的信条里,公平是使世界树能够生生不息.持续运转的根本基石.世界树的形态可以用一个数学模型来描述:世界树中有n个种族,种

【环套树+树形dp】Bzoj1040 [ZJOI2008] 骑士

Description Z国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英.他们劫富济贫,惩恶扬善,受到社会各界的赞扬.最近发生了一件可怕的事情,邪恶的Y国发动了一场针对Z国的侵略战争.战火绵延五百里,在和平环境中安逸了数百年的Z国又怎能抵挡的住Y国的军队.于是人们把所有的希望都寄托在了骑士团的身上,就像期待有一个真龙天子的降生,带领正义打败邪恶.骑士团是肯定具有打败邪恶势力的能力的,但是骑士们互相之间往往有一些矛盾.每个骑士都有且仅有一个自己最厌恶的骑士(当然不是他自己),他是绝对不

luogu2607/bzoj1040 [ZJOI2008]骑士 (基环树形dp)

N个点,每个点发出一条边,那么这个图的形状一定是一个基环树森林(如果有重边就会出现森林) 那我做f[0][x]和f[1][x]分别表示对于x子树,x这个点选还是不选所带来的最大价值 然后就变成了这好几个环上不能选相邻的点,最大的价值和 我们把这个环从N到1处断开,然后钦定一下1选还是不选,统计一下答案就可以了. 1 #include<bits/stdc++.h> 2 #define pa pair<int,int> 3 #define ll long long 4 using na

[BZOJ1040][ZJOI2008]骑士(环套树dp)

1040: [ZJOI2008]骑士 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 5816  Solved: 2263[Submit][Status][Discuss] Description Z国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英.他们劫富济贫,惩恶扬善,受到社会各 界的赞扬.最近发生了一件可怕的事情,邪恶的Y国发动了一场针对Z国的侵略战争.战火绵延五百里,在和平环境 中安逸了数百年的Z国又怎能抵挡的住Y国的军队.于是人

BZOJ 1040:[ZJOI2008]骑士(环套外向树+树形DP)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1040 [题目大意] 给出环套外向树森林,求最大权独立集. [题解] 我们对于每个连通块,找到环上的一条边拆开,对于边的两端分别做树形DP, 假设两端点位x和y,那么不包含x的dp值涵盖了是否包含y两种情况, 同理,以y为根的也是,因为边的两端不能同时取到,因此对于两者取最大值即可. 代码中f[x]表示包含x的dp值,g[x]表示不包含x的dp值. [代码] #include <cst

bzoj1040(ZJOI2008)骑士——基环树

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1040 基环树的模板. 套路就是把环断开,先把一端作为根节点,强制不选:再把另一端作为根节点,强制不选. 人家的这个判断环的方法真好!还顺便没有连上环的那条边,省下了在函数里判断. 别忘了有多棵基环树! #include<iostream> #include<cstdio> #include<cstring> #define ll long long using n

BZOJ 1040 ZJOI 2008 骑士 基环树林+树形DP

题目大意:有一些骑士,他们每个人都有一个权值.但是由于一些问题,每一个骑士都特别讨厌另一个骑士.所以不能把他们安排在一起.求这些骑士所组成的编队的最大权值和是多少. 思路:首先貌似是有向图的样子,但是一个人讨厌另一个人,他们两个就不能在一起,所以边可以看成是无向的. n个点,n条无向边,好像是一颗基环树.但其实这是一个基环树林,因为题中并没有说保证图一定联通. 然后就可以深搜了,处理出每一个联通块.其实每一个联通块就是一个基环树,在这个基环树上进行树形DP.求出最大值,然后累加到答案上.答案要开