BZOJ_1040_[ZJOI2008]骑士_树形DP

题意:

Z国的骑士团是一个很有势力的组织,帮会中汇聚了来自各地的精英。他们劫富济贫,惩恶扬善,受到社会各

界的赞扬。最近发生了一件可怕的事情,邪恶的Y国发动了一场针对Z国的侵略战争。战火绵延五百里,在和平环境

中安逸了数百年的Z国又怎能抵挡的住Y国的军队。于是人们把所有的希望都寄托在了骑士团的身上,就像期待有一

个真龙天子的降生,带领正义打败邪恶。骑士团是肯定具有打败邪恶势力的能力的,但是骑士们互相之间往往有一

些矛盾。每个骑士都有且仅有一个自己最厌恶的骑士(当然不是他自己),他是绝对不会与自己最厌恶的人一同出

征的。战火绵延,人民生灵涂炭,组织起一个骑士军团加入战斗刻不容缓!国王交给了你一个艰巨的任务,从所有

的骑士中选出一个骑士军团,使得军团内没有矛盾的两人(不存在一个骑士与他最痛恨的人一同被选入骑士军团的

情况),并且,使得这支骑士军团最具有战斗力。为了描述战斗力,我们将骑士按照1至N编号,给每名骑士一个战

斗力的估计,一个军团的战斗力为所有骑士的战斗力总和。

分析:

可以发现图是由若干个连通块组成,每个连通块都是一个基环树。

我的做法是标记一下每个连通块有没有环。

如果有环则拆环,做两遍树形DP。

代码(丑):

#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 1000050
#define LL long long
int head[N],to[N<<1],nxt[N<<1],cnt,n;
int fa[N],root,son,rt[N],so[N],lian[N];
LL f[N],g[N],val[N];
inline void add(int u,int v){
    to[++cnt]=v;nxt[cnt]=head[u];head[u]=cnt;
}
int find(int x){
    return fa[x]==x?x:fa[x]=find(fa[x]);
}
void dfs3(int x,int y){
    f[x]=val[x];
    for(int i=head[x];i;i=nxt[i]){
        if(to[i]!=y){
            dfs3(to[i],x);
            f[x]+=g[to[i]];
            g[x]+=max(f[to[i]],g[to[i]]);
            f[to[i]]=g[to[i]]=0;
        }
    }
}
void dfs1(int x,int y){
    if(x!=son)f[x]=val[x];
    for(int i=head[x];i;i=nxt[i]){
        if(to[i]!=y){
            dfs1(to[i],x);
            if(x!=son)
                f[x]+=g[to[i]];
            if(to[i]==son)g[x]+=g[to[i]];
            else g[x]+=max(f[to[i]],g[to[i]]);
            f[to[i]]=g[to[i]]=0;
        }
    }
}
void dfs2(int x,int y){
    if(x!=root)f[x]=val[x];
    for(int i=head[x];i;i=nxt[i]){
        if(to[i]!=y){
            dfs2(to[i],x);
            if(x!=root)
                f[x]+=g[to[i]];
            if(to[i]==root)g[x]+=g[to[i]];
            else g[x]+=max(g[to[i]],f[to[i]]);
            f[to[i]]=g[to[i]]=0;
        }
    }
}
int main(){
    scanf("%d",&n);
    for(int i=1;i<=n;i++)fa[i]=i;
    int x;
    for(int i=1;i<=n;i++){
        scanf("%lld%d",&val[i],&x);
        int di=find(i),dx=find(x);
        if(di!=dx){
            fa[di]=dx;lian[dx]|=lian[di];
            add(i,x);add(x,i);
        }
        else{
            rt[di]=rt[dx]=i;
            so[di]=so[dx]=x;
            lian[di]=lian[dx]=1;
        }
    }
    LL ans=0;
    for(int i=1;i<=n;i++){
        if(fa[i]==i){
            if(!lian[i]){
                dfs3(i,0);
                ans+=max(f[i],g[i]);
            }else{
                root=rt[i],son=so[i];
                dfs1(root,0);
                LL now=max(f[root],g[root]);
                f[root]=g[root]=0;
                dfs2(son,0);
                ans+=max(max(now,f[son]),g[son]);
            }
        }
    }
    printf("%lld\n",ans);
}

原文地址:https://www.cnblogs.com/suika/p/8476370.html

时间: 2024-08-30 00:57:32

BZOJ_1040_[ZJOI2008]骑士_树形DP的相关文章

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

算法提高 金属采集_树形dp

算法提高 金属采集 时间限制:1.0s   内存限制:256.0MB 问题描述 人类在火星上发现了一种新的金属!这些金属分布在一些奇怪的地方,不妨叫它节点好了.一些节点之间有道路相连,所有的节点和道路形成了一棵树.一共有 n 个节点,这些节点被编号为 1~n .人类将 k 个机器人送上了火星,目的是采集这些金属.这些机器人都被送到了一个指定的着落点, S 号节点.每个机器人在着落之后,必须沿着道路行走.当机器人到达一个节点时,它会采集这个节点蕴藏的所有金属矿.当机器人完成自己的任务之后,可以从任

【BZOJ】1040: [ZJOI2008]骑士 环套树DP

[题意]给定n个人的ai和bi,表示第i个人能力值为ai且不能和bi同时选择,求能力值和最大的选择方案.n<=10^6. [算法]环套树DP(基环树) [题解]n个点n条边--基环森林(若干环套树子图). 若原图是树,经典DP做法:f[i][0/1]表示i点选或不选的最大能力值和,则f[i][0]=Σmax{f[j][0],f[j][1]},f[i][1]=Σf[j][0]+a[i],j=son[i]. 找环:dfs到访问过的点,标记环上的一条边. 破环:和普通树上DP唯一的区别是,标记边两端不

BZOJ_1060_时态同步_树形DP

题意:http://www.lydsy.com/JudgeOnline/problem.php?id=1060 分析:水水的树形DP. 用儿子的最大值更新父亲,边更新边累加ans. 代码: #include <stdio.h> #include <string.h> #include <algorithm> using namespace std; #define N 1000010 #define LL long long int head[N],to[N<&l

[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_4726_[POI2017]Sabota?_树形DP

Description 某个公司有n个人, 上下级关系构成了一个有根树.其中有个人是叛徒(这个人不知道是谁).对于一个人, 如果他 下属(直接或者间接, 不包括他自己)中叛徒占的比例超过x,那么这个人也会变成叛徒,并且他的所有下属都会变 成叛徒.你要求出一个最小的x,使得最坏情况下,叛徒的个数不会超过k. Input 第一行包含两个正整数n,k(1<=k<=n<=500000). 接下来n-1行,第i行包含一个正整数p[i+1],表示i+1的父亲是p[i+1](1<=p[i+1]&

[bzoj1812][IOI2006]riv_多叉树转二叉树_树形dp

riv bzoj-1812 IOI-2006 题目大意:给定一棵n个点树,要求在上面建立k个收集站.点有点权,边有边权,整棵树的代价是每个点的点权乘以它和它的最近的祖先收集站的距离积的和. 注释:$1\le n \le 100$,$1\le k \le 50$. 想法:显然,这是一道树形dp题.状态也非常容易... ...只不过,我们好像要枚举子集... 所以,我们这里有一个黑科技:多叉树转二叉树. 我们先把它转成二叉树,然后再进行转移即可. 状态:dp[i][j][k]表示以i为根的子树中选出

[bzoj2657][Zjoi2012]旅游 journey_ 对偶图_树形dp

旅游 bzoj-2657 Zjoi-2012 题目大意:题目链接 注释:$1\le K\le 2\cdot 10^5$. 想法:这题... 感觉和上一个题的提示有些类似,就是题目生怕你不知道这是一道对偶图的题... ... 我们先把它转成对偶图.然后我们只把分割n变形的线段两侧的点之间连边,这样就是一棵树. 紧接着我们想要遍历最多的城市,其实就是找树上直径.树形dp即可. 最后,附上丑陋的代码... ... #include <iostream> #include <cstdio>

【bzoj1040】【zjoi2008】骑士(树形dp)

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