CF1249F Maximum Weight Subset
CF1249F Maximum Weight Subset
题目描述
You are given a tree, which consists of nn vertices. Recall that a tree is a connected undirected graph without cycles.
Example of a tree.Vertices are numbered from 11 to nn . All vertices have weights, the weight of the vertex vv is a_va**v .
Recall that the distance between two vertices in the tree is the number of edges on a simple path between them.
Your task is to find the subset of vertices with the maximum total weight (the weight of the subset is the sum of weights of all vertices in it) such that there is no pair of vertices with the distance kk or less between them in this subset.
输入格式
The first line of the input contains two integers nn and kk ( 1 \le n, k \le 2001≤n,k≤200 ) — the number of vertices in the tree and the distance restriction, respectively.
The second line of the input contains nn integers a_1, a_2, \dots, a_na1,a2,…,a**n ( 1 \le a_i \le 10^51≤a**i≤105 ), where a_ia**i is the weight of the vertex ii .
The next n - 1n−1 lines contain edges of the tree. Edge ii is denoted by two integers u_iu**i and v_iv**i — the labels of vertices it connects ( 1 \le u_i, v_i \le n1≤u**i,v**i≤n , u_i \ne v_iu**i≠v**i ).
It is guaranteed that the given edges form a tree.
输出格式
Print one integer — the maximum total weight of the subset in which all pairs of vertices have distance more than kk .
题意翻译
给定一棵含nn个节点的树,每个节点含一个点权a[i]a[i]。现在请你选出一些节点,使得这些节点的权值和最大并且这些节点中任意两个节点的距离都>k>k。并输出这个最大的权值。
输入第一行含两个整数n,kn,k,之后是nn个整数a[i]a[i],之后是n-1n−1行,每行两个整数,描述树的每条边。
输入输出样例
输入 #1复制
输出 #1复制
输入 #2复制
输出 #2复制
题解:
题目翻译过来发第一篇题解(英语巨菜无所谓)
来讲一下正解:树上DP。
我们从0开始为节点编号,并设置0为根节点。
DP的状态是这样的:设\(dp[x][dep]\)为以\(x\)为根节点的子树中所有选中节点中深度最小的节点深度为\(dep\)时的最大价值(有点绕...请好好理解)
这里要注意,我们定义的深度并不是整棵树的深度。因为动态规划是利用状态转移,所以我们定义的这个\(dep\)是在当前状态下的深度。通俗地讲,就是这棵子树的深度(子树根节点深度为0)。
那么我们开始考虑状态转移。这道题的一个限定条件(比较难搞)还有一个任意两点的距离都大于\(k\)。我们的状态转移(包括状态设置)都是考虑这个而进行的。
可以看出,对于每一个节点,它既统计了它自己的答案,又对它的合法的祖宗们(滑稽.jpg)的答案有贡献。所以,据此,我们考虑两种情况:第一种,当前的节点是子树的根,这个时候它是它的小家族的最大的祖宗,开始统计它的答案。第二种,当前的节点是儿子或孙儿或...这个时候它要对它上面的祖宗们贡献答案。
根据我们对状态的定义,在第一种情况下,当前节点的深度为0,那么当前节点的答案
\]
在第二种情况下,我们则需要遍历\(x\)节点的所有儿子,然后挑出对答案贡献最大的儿子来更新答案。转移方程就是:
\]
这个时候就能比较容易地想到用深搜解决问题。因为深搜的性质恰好满足:从上搜到低,在从低到上更新答案的过程。所以,我们把答案再从\(n-1\)号点跑回\(0\)号点即可。最终的答案就是\(dp[0][0]\)。
代码:
#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=210;
int n,k;
int a[maxn],d[maxn],dp[maxn][maxn],tmp[maxn];
int tot,head[maxn],nxt[maxn<<1],to[maxn<<1];
void add(int x,int y)
{
	to[++tot]=y;
	nxt[tot]=head[x];
	head[x]=tot;
}
void dfs(int x,int f)
{
	dp[x][0]=a[x];
	for(int i=head[x];i;i=nxt[i])
	{
		int y=to[i];
		if(y!=f)
			dfs(y,x);
	}
	for(int i=0;i<n;i++)
	{
		if(!i)
		{
			for(int l=head[x];l;l=nxt[l])
			{
				int y=to[l];
				if(y==f)
					continue;
				dp[x][0]+=dp[y][k];
			}
		}
		else
		{
			for(int l=head[x];l;l=nxt[l])
			{
				int y=to[l];
				if(y==f)
					continue;
				int now=dp[y][i-1];
				for(int j=head[x];j;j=nxt[j])
				{
					int yy=to[j];
					if(yy==f || yy==y)
						continue;
					now+=dp[yy][max(i-1,k-i)];
				}
				dp[x][i]=max(dp[x][i],now);
			}
		}
	}
	for(int i=n;i>=1;i--)
		dp[x][i-1]=max(dp[x][i-1],dp[x][i]);
}
int main()
{
	scanf("%d%d",&n,&k);
	for(int i=0;i<n;i++)
		scanf("%d",&a[i]);
	for(int i=0;i<n-1;i++)
	{
		int u,v;
		scanf("%d%d",&u,&v);
		u--,v--;
		add(u,v);
		add(v,u);
	}
	dfs(0,-1);
	printf("%d",dp[0][0]);
	return 0;
}
												
											CF1249F Maximum Weight Subset的更多相关文章
- 【CF1249F】Maximum Weight Subset(贪心)
		
题意:给定一棵n个点带点权的树,要求从中选出一个点集,使得这些点两两之间距离都大于K,求最大点权和 n,K<=2e2,1<=a[i]<=1e5 思路:树形DP显然可做,极限是n方,然 ...
 - codeforces#1249F. Maximum Weight Subset(树上dp)
		
题目链接: http://codeforces.com/contest/1249/problem/F 题意: 一棵树的每个节点有个权值,选择一个节点集,使得任意点对的距离大于$k$ 求最大节点集权值, ...
 - Codeforces 1249 F. Maximum Weight Subset
		
传送门 设 $f[x][i]$ 表示 $x$ 的子树中,离 $x$ 最近的选择的节点距离为 $i$ 的合法方案的最大价值 设 $val[x]$ 表示节点 $x$ 的价值,首先有 $f[x][0]=va ...
 - Codeforces 1249F Maximum Weight Subset (贪心)
		
题意 在一颗有点权的树上,选若干个点,使得这些点两两距离大于k,且点权和最大 思路 贪心的取比较大的值即可 将所有点按照深度从大到小排序,如果当前点点权\(a[i]\)大于0,则将距离为k以内的所有点 ...
 - F. Maximum Weight Subset(贪心or树形dp解法)
		
题:https://codeforces.com/contest/1249/problem/F 题意:给一颗树,边权为1,节点有点权,问取到一个点集,俩俩之间路径超过k,是点权和最大 思路:贪心地取点 ...
 - some problem
		
CF1257F Make Them Similar $solution:$ 折半搜索后考虑如何维护两个数组的和,可以将 $A$ 中每个数减 $A_1$ ,$B$ 中每个数被减 $B_1$ ,$map$ ...
 - Codeforces Round #595 (Div. 3)
		
A - Yet Another Dividing into Teams 题意:n个不同数,分尽可能少的组,要求组内没有两个人的差恰为1. 题解:奇偶分组. int a[200005]; void te ...
 - CF-595
		
题目传送门 A .Yet Another Dividing into Teams sol:原先是用比较复杂的方法来解的,后来学弟看了一眼,发现不是1就是2,当出现两个人水平相差为1就分成两组,1组全是 ...
 - Codeforces 828F Best Edge Weight - 随机堆 - 树差分 - Kruskal - 倍增算法
		
You are given a connected weighted graph with n vertices and m edges. The graph doesn't contain loop ...
 
随机推荐
- 基于阿里云平台的使用python脚本发送短信
			
第一步:点击短信服务下的帮助文档 第二步:安装python的SDK:点击安装python sdk 第三步:直接通过python的pip工具安装即可,方便快捷: 第四步:点击红框进行测试: 第五步:测试 ...
 - html和css常见问题解答
			
1. 详细描述层叠和继承的概念. 元素内嵌样式(用元素的全局属性style定义的样式) 文档内嵌样式(定义在style元素中的样式) 外部样式(用link元素导入的样式) 用户样式(用户定义的样式) ...
 - Java面试题的个人总结
			
面试总结 第一轮:电话初面 第二轮:技能面谈[技能职位尽量避免多谈处理上的作业] 第三轮:高管复试 第四轮:HR终究供认 一面:首要供认对阿里的意向度(假定异地更会考虑对作业地址(杭州)的意向度!阿里 ...
 - 通过ES6 Module看import和require区别
			
前言 说到import和require,大家平时开发中一定不少见,尤其是需要前端工程化的项目现在都已经离不开node了,在node环境下这两者都是大量存在的,大体上来说他们都是为了实现JS代码的模块化 ...
 - jwt工具类的封装
			
第一步先引入jar: <dependency> <groupId>com.nimbusds</groupId> <artifactId>nimbus-j ...
 - python自动登录代码
			
公司有很多管理平台,账号有禁用机制,每个月至少登录一次,否则禁用.导致有时候想登录某个平台的时候,发现账号已经被禁用了,还得走流程解禁.因此用python实现了一下自动登录,每天定时任务运行一次.ps ...
 - Docker是什么、为什么是一种趋势
			
Docker的思想来自于集装箱,集装箱解决了什么问题?在一艘大船上,可以把货物规整的摆放起来.并且各种各样的货物被集装箱标准化了,集装箱和集装箱之间不会互相影响.那么我就不需要专门运送水果的船和专门运 ...
 - 用Python在Excel里画出蒙娜丽莎
			
前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: 麦麦麦造 PS:如有需要Python学习资料的小伙伴可以加点击下方链 ...
 - PHP面试题2019年滴滴出行工程师面试题及答案解析
			
一.单选题(共30题,每题5分) 1.下列关于PHP垃圾回收的说法,错误的是? A.开启/关闭垃圾回收机制可以通过修改php配置实现 B.可以在程序中使用gc_enable() 和 gc_disabl ...
 - C#在循环中使用Random时生成的随机数相同的解决办法
			
场景 在循环中使用 Random y = new Random(); 生成随机数时每次循环生成的数是一样的. ; i < ;i++ ) { Random y = new Random(); Po ...