【换根DP例题】——P3047 [USACO12FEB] Nearby Cows G


题目

原题链接

描述

Farmer John has noticed that his cows often move between nearby fields. Taking this into account, he wants to plant enough grass in each of his fields not only for the cows situated initially in that field, but also for cows visiting from nearby fields.
Specifically, FJ’s farm consists of N fields (1 <= N <= 100,000), where some pairs of fields are connected with bi-directional trails (N-1 of them in total). FJ has designed the farm so that between any two fields i and j, there is a unique path made up of trails connecting between i and j. Field i is home to C(i) cows, although cows sometimes move to a different field by crossing up to K trails (1 <= K <= 20).
FJ wants to plant enough grass in each field i to feed the maximum number of cows, M(i), that could possibly end up in that field – that is, the number of cows that can potentially reach field i by following at most K trails. Given the structure of FJ’s farm and the value of C(i) for each field i, please help FJ compute M(i) for every field i.

给你一棵 n 个点的树,点带权,对于每个节点求出距离它不超过 k 的所有节点权值和 m i。

输入格式

  • Line 1: Two space-separated integers, N and K.

  • Lines 2…N: Each line contains two space-separated integers, i and j ( 1 < = i , j < = N 1 <= i,j <= N 1<=i,j<=N) indicating that fields i and j are directly connected by a trail.

  • Lines N+1…2N: Line N+i contains the integer C(i). ( 0 < = C ( i ) < = 1000 0 <= C(i) <= 1000 0<=C(i)<=1000)

  • 第一行两个正整数n,k。

  • 接下来n−1 行,每行两个正整数 u,v,表示 u,v 之间有一条边。

  • 最后 n 行,每行一个非负整数 c i,表示点权。

输出格式

  • Lines 1…N: Line i should contain the value of M(i).

  • 输出 n 行,第 i 行一个整数表示 m i。

输入/输出例子

输入

6 2
5 1
3 6
2 4
2 1
3 2
1
2
3
4
5
6

输出

15
21
16
10
8
11

提示

【数据范围】
对于 100%的数据: 1 ≤ n ≤ 100000 1≤n≤100000 1n100000 1 ≤ k ≤ 20 1≤k≤20 1k20 0 ≤ c i ≤ 1000 0≤c_{i}≤1000 0ci1000


解题思路

分析

考虑暴力方法:每个点都进行一次树形DP,时间复杂度 O ( n 2 ) O(n^2) O(n2)
优化:对于根节点切换时,我们需要用 O ( 1 ) O(1) O(1)的时间复杂度算出另一个根的数据,所以用换根DP

初始化/第一遍 d f s dfs dfs

定义:
f [ i ] [ j ] f[i][j] f[i][j]表示以 i i i为根节点时整棵树中距离 i i i不超过 j j j的节点数
g [ i ] [ j ] g[i][j] g[i][j]表示以 i i i为根节点的子树中距离 i i i j j j的节点数。
所以我们可以得到:
g [ u ] [ k ] + = g [ t o ] [ k − 1 ] g[u][k]+=g[to][k-1] g[u][k]+=g[to][k1]

code1

void dfs(int u,int fa)
{
	for(int i=head[u];i;i=edge[i].next)
	{
		int w=edge[i].w,to=edge[i].v;
		if(to==fa)
			continue;
		dfs(to,u);
		foru(j,1,k)
			g[u][j]+=g[to][j-1];
	}
}

d f s dfs dfs结束后,我们再对 g g g做前缀和,那么现在 g [ i ] [ j ] g[i][j] g[i][j]就表示 i i i为根节点的子树中距离 i i i不超过 j j j的节点个数
同时,由于我们第一遍 d f s dfs dfs是以 1 1 1为根节点的,所以: f [ 1 ] [ i ] = g [ 1 ] [ i ] f[1][i]=g[1][i] f[1][i]=g[1][i]

code2

	f[1][0]=g[1][0];
	foru(i,1,n)
	{
		foru(j,1,20)
			g[i][j]+=g[i][j-1];
		f[1][i]=g[1][i];
	}

动态转移方程/第二遍 d f s dfs dfs

我们已经知道了 f [ 1 ] [ i ] f[1][i] f[1][i],接下里需要转移 f [ j ] [ i ] f[j][i] f[j][i]

在这里插入图片描述
如图, f [ v ] [ k ] = g [ v ] [ k ] + A f[v][k]=g[v][k]+A f[v][k]=g[v][k]+A
(A代表A中距离v不超过k的点的点数)
A = f [ j ] [ k − 1 ] − g [ v ] [ k − 2 ] A=f[j][k-1]-g[v][k-2] A=f[j][k1]g[v][k2]
注:这个可以自己推一下

code

void dfs_(int u,int fa)
{
	for(int i=head[u];i;i=edge[i].next)
	{
		int w=edge[i].w,to=edge[i].v;
		if(to==fa)
			continue;
		f[to][0]=g[to][0];
		f[to][1]=g[to][1]+f[u][0];
		foru(j,2,k)
			f[to][j]=g[to][j]+f[u][j-1]-g[to][j-2];
		dfs_(to,u);
	}
}

AC Code

#include<bits/stdc++.h>
#define mod 1
#define int long long
#define foru(i,a,b) for(register int i=a;i<=b;i++)
#define ford(i,a,b) for(register int i=a;i>=b;i--)
using namespace std;
const int N=1e5+1;
int n,k,g[N][21],f[N][21],head[N],idx,x,y;
struct fy
{
	int v,w,next;
}edge[N<<1];
void add(int x,int y,int z)
{
	edge[++idx].v=y,edge[idx].w=z,edge[idx].next=head[x],head[x]=idx;
}
inline char gc()
{
  	static char now[1<<20],*S,*T;
  	if(T==S)
  	{
		T=(S=now)+fread(now,1,1<<20,stdin);
    	if(T==S) 
			return EOF;
  	}
  return *S++;
}
template <typename T>
inline void Read(T&x) 
{
  	x=0;
  	char c=gc();
  	while(c<'0'||c>'9') 
  		c=gc();
  	x=c-'0';
  	while((c=gc())>='0'&&c<='9') 
  		x=x*10+c-'0';
}
template <typename T, typename... Args>
inline void Read(T&x,Args&...args)
{
  	Read(x);
  	Read(args...);
}
void dfs(int u,int fa)
{
	for(int i=head[u];i;i=edge[i].next)
	{
		int w=edge[i].w,to=edge[i].v;
		if(to==fa)
			continue;
		dfs(to,u);
		foru(j,1,k)
			g[u][j]+=g[to][j-1];
	}
}
void dfs_(int u,int fa)
{
	for(int i=head[u];i;i=edge[i].next)
	{
		int w=edge[i].w,to=edge[i].v;
		if(to==fa)
			continue;
		f[to][0]=g[to][0];
		f[to][1]=g[to][1]+f[u][0];
		foru(j,2,k)
			f[to][j]=g[to][j]+f[u][j-1]-g[to][j-2];
		dfs_(to,u);
	}
}
signed main()
{
	Read(n,k);
	foru(i,2,n)
		Read(x,y),add(x,y,1),add(y,x,1);
	foru(i,1,n)
		Read(g[i][0]);
	dfs(1,0);
	f[1][0]=g[1][0];
	foru(i,1,n)
	{
		foru(j,1,20)
			g[i][j]+=g[i][j-1];
		f[1][i]=g[1][i];
	}
	dfs_(1,0);
	foru(i,1,n)
		printf("%lld\n",f[i][k]);
}

更多方法

更多方法

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值