基环树DP
一般的树形DP都是满足N个点N-1条边的数。可是有一类问题是N个点N条边,这样一定会有环,这样的图就叫基环树。
对于这类题,我们可以先dfs一次,找出环,然后把环上的一条边删除。那么图就能变成树,套用树规很容易解决。
假设将要删去的环上的边为E,边上两端点为u,v。然后分别以u和v为树根做一次DP,去最优值就可以了。
Tips:在存图是,可将cnt的初始值设为1,即第一条边的编号设为2,然后用类似于网络流中存储反向边的方法来做。
模板题:
BZOJ 1040[ZJOI2008]骑士
/*
Author:Wdvxdr
Problem:Bzoj 1040
Time:2017/12/25
*/
#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e6+10;
struct node{int v,nxt;}e[maxn<<1];
int head[maxn],cnt=1,a[maxn],n,Rtree,Ltree,DelEdge;
bool vis[maxn];
long long f[maxn][2];
inline void add(int u,int v)
{
e[++cnt] = (node){v,head[u]};
head[u]=cnt;
}
void dfs(int u,int fa)
{
vis[u] = 1;
for(int i=head[u];i;i=e[i].nxt)
{
int v = e[i].v;
if(v!=fa)
if(!vis[v])
{
dfs(v,u);
}
else
Rtree = u,Ltree = v,DelEdge=i;
}
}
void dp(int u,int fa)
{
int v;
f[u][0]=0;f[u][1]=a[u];
for(int i=head[u];i;i=e[i].nxt)
{
if(i==DelEdge||i==(DelEdge^1)) continue;
v = e[i].v;
if(v!=fa)
{
dp(v,u);
f[u][0] += max(f[v][0],f[v][1]);
f[u][1] += f[v][0];
}
}
}
int main()
{
int t;long long temp,ans=0;
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&a[i],&t);
add(t,i);add(i,t);
}
for(int i=1;i<=n;i++)
{
if(vis[i]) continue;
dfs(i,0);
dp(Rtree,0);
temp = f[Rtree][0];
dp(Ltree,0);
temp = max(temp,f[Ltree][0]);
ans += temp;
}
cout << ans;
return 0;
}

浙公网安备 33010602011771号