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**in , u_i \ne v_iu**iv**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,那么当前节点的答案

\[dp[x][dep]=a[x]+\sum_{to\in children(x)}dp[to][max(0,k-dep)].
\]

在第二种情况下,我们则需要遍历\(x\)节点的所有儿子,然后挑出对答案贡献最大的儿子来更新答案。转移方程就是:

\[dp[x][dep]=max(dp[x][dep],dp[x][dep-1]+\sum_{to\in children(x)}dp[to][max(dep-1,k-dep)]).
\]

这个时候就能比较容易地想到用深搜解决问题。因为深搜的性质恰好满足:从上搜到低,在从低到上更新答案的过程。所以,我们把答案再从\(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的更多相关文章

  1. 【CF1249F】Maximum Weight Subset(贪心)

    题意:给定一棵n个点带点权的树,要求从中选出一个点集,使得这些点两两之间距离都大于K,求最大点权和 n,K<=2e2,1<=a[i]<=1e5 思路:树形DP显然可做,极限是n方,然 ...

  2. codeforces#1249F. Maximum Weight Subset(树上dp)

    题目链接: http://codeforces.com/contest/1249/problem/F 题意: 一棵树的每个节点有个权值,选择一个节点集,使得任意点对的距离大于$k$ 求最大节点集权值, ...

  3. Codeforces 1249 F. Maximum Weight Subset

    传送门 设 $f[x][i]$ 表示 $x$ 的子树中,离 $x$ 最近的选择的节点距离为 $i$ 的合法方案的最大价值 设 $val[x]$ 表示节点 $x$ 的价值,首先有 $f[x][0]=va ...

  4. Codeforces 1249F Maximum Weight Subset (贪心)

    题意 在一颗有点权的树上,选若干个点,使得这些点两两距离大于k,且点权和最大 思路 贪心的取比较大的值即可 将所有点按照深度从大到小排序,如果当前点点权\(a[i]\)大于0,则将距离为k以内的所有点 ...

  5. F. Maximum Weight Subset(贪心or树形dp解法)

    题:https://codeforces.com/contest/1249/problem/F 题意:给一颗树,边权为1,节点有点权,问取到一个点集,俩俩之间路径超过k,是点权和最大 思路:贪心地取点 ...

  6. some problem

    CF1257F Make Them Similar $solution:$ 折半搜索后考虑如何维护两个数组的和,可以将 $A$ 中每个数减 $A_1$ ,$B$ 中每个数被减 $B_1$ ,$map$ ...

  7. Codeforces Round #595 (Div. 3)

    A - Yet Another Dividing into Teams 题意:n个不同数,分尽可能少的组,要求组内没有两个人的差恰为1. 题解:奇偶分组. int a[200005]; void te ...

  8. CF-595

    题目传送门 A .Yet Another Dividing into Teams sol:原先是用比较复杂的方法来解的,后来学弟看了一眼,发现不是1就是2,当出现两个人水平相差为1就分成两组,1组全是 ...

  9. 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 ...

随机推荐

  1. 201871010114-李岩松《面向对象程序设计(java)》第十四周学习总结

    项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p ...

  2. 2019-2020-1 20199305《Linux内核原理与分析》第六周作业

    系统调用的三层机制(下) (一)给MenuOS增加命令 (1)打开虚拟机,首先用rm -rf menu指令删除当前的menu目录,然后用git clone重新克隆一个新版本的menu,进入menu,运 ...

  3. docker /var/lib/docker/aufs/mnt 目录满了怎么清理

    1.创建脚本文件 vi cleandocker.sh 内容如下: #!/bin/sh echo "==================== start clean docker contai ...

  4. luogu5823 课表的排列

    题目链接 problem 构造一个长度为\(2n\)的数列.满足: 1.[1,n]中每个数字恰好出现两次. 2.将所有相同数字之间相隔的数字个数排序后,得到公差为1的等差数列. 保证n为奇数. sol ...

  5. Mysql 免安装教程 百度云网盘有文件和word说明

    一 把文件解压到一个目录下 这是解压后的目录 将my.ini文件考进去 双击打开my.ini 找到这两行更改成自己的解压路径保存 右键此电脑属性 找到高级系统设置配置环境变量 新建—>变量值是解 ...

  6. JMS消息传递类型特点介绍

    对于消息的传递有两种类型: 一种是点对点的,即一个生产者和一个消费者一一对应: 另一种是发布/ 订阅模式,即一个生产者产生消息并进行发送后,可以由多个消费者进 行接收. 特点介绍: 点到点模型点对点传 ...

  7. jQuery 源码分析(十七) 事件系统模块 实例方法和便捷方法 详解

    实例方法和便捷方法是指jQuery可以直接通过链接操作的方法,是通过调用$.event上的方法(上一节介绍的底层方法)来实现的,常用的如下: on(types,selector,data,fn,one ...

  8. 【HDU6037】Expectation Division(动态规划,搜索)

    [HDU6037]Expectation Division(动态规划,搜索) 题面 Vjudge 你有一个数\(n\),\(n\le 10^{24}\),为了方便会告诉你\(n\)分解之后有\(m\) ...

  9. KiRaiseException函数逆向

    KiRaiseException函数是记录异常的最后一步,在这之后紧接着就调用KiDispatchException分发异常. 我们在逆向前,先看一下书中的介绍: 1. 概念认知: KiRaiseEx ...

  10. python中反转列表的三种方式

    1.内建函数reversed() li =[1, 2, 3, 4, 5, 6] a = list(reversed(li)) print (a) 注意:reversed()函数返回的是一个迭代器,而不 ...