题目链接

看起来似乎跟最小点覆盖有点像。但区别在于:

最小点覆盖要求所有边在其中,而本题要求所有点在其中。

即:一个点不选时,它的儿子不一定需要全选。

画图理解:



对于这样一幅图,本题中可以这样选择:3,53,53,5:

333将相邻的点2,42,42,4覆盖,而555将相邻的点1,61,61,6覆盖,因此所有点都被覆盖了。

那么就必须修改状态了。

Dynamic Programing

考虑对于一个点,如果它被覆盖了,只有三种可能:

  1. 自己被标记
  2. 父亲被标记
  3. 儿子被标记

因此我们设计状态:

  1. dp[now][0]dp[now][0]dp[now][0]为自己被标记
  2. dp[now][1]dp[now][1]dp[now][1]为父亲被标记
  3. dp[now][2]dp[now][2]dp[now][2]为儿子被标记

那么如何转移呢?

自己被标记时,儿子的3种状态都是合法的,因此有:

dp[now][0]=∑min⁡(dp[son][0],dp[son][1],dp[son][2])dp[now][0] = \sum \min(dp[son][0],dp[son][1],dp[son][2])dp[now][0]=∑min(dp[son][0],dp[son][1],dp[son][2])

而父亲被标记时,儿子不能被父亲标记了,只能自己标记或者被儿子的儿子标记。

dp[now][1]=∑min⁡(dp[son][0],dp[son][2])dp[now][1] = \sum \min(dp[son][0],dp[son][2])dp[now][1]=∑min(dp[son][0],dp[son][2])

被儿子标记的情况就复杂一些。首先,被儿子标记时,所有儿子肯定无法被父亲标记,因此首先有:

dp[now][2]=∑min⁡(dp[son][0],dp[son][2])dp[now][2] = \sum \min(dp[son][0],dp[son][2])dp[now][2]=∑min(dp[son][0],dp[son][2])

但是至少需要有一个儿子标记自己,才能让当前节点被儿子控制。如果一遍下来都没有选取dp[son][0]dp[son][0]dp[son][0]的情况怎么办呢?

我们考虑一个儿子,如果它被儿子标记更优,我们却强制它标记自己,那么代价就是dp[son][0]−dp[son][2]dp[son][0] - dp[son][2]dp[son][0]−dp[son][2]。

那么我们在遍历儿子时记录,看是否有儿子选取了控制自己,如果没有的话,就选择代价最小的那个儿子强制标记它即可。

Code

实现似乎没有太多坑点。

但要注意:当一个点是叶子结点时,dp[now][2]dp[now][2]dp[now][2]即被儿子标记的代价一定是+∞+\infty+∞,在代码中我直接并在了最后的处理中。可以感受一下。

另外就是这题的数据问题,虽然说题面是严格父亲对应儿子,但我建单向边死活过不了第三个点,改成双向边后AC。

最后答案,根节点不能被父亲控制。

#include <cstdio>
#include <cstring>
using namespace std;
template<typename T>
void read(T &r)
{
static char c; r=0;
for(c=getchar();c>'9'||c<'0';c=getchar());
for(;c>='0'&&c<='9';r=(r<<1)+(r<<3)+(c^48),c=getchar());
}
struct node
{
int to, next;
node() {}
node(const int &_to, const int &_next) : to(_to), next(_next) {}
} lines[3002];
int head[1501];
void add(const int &x, const int &y)
{
static int tot = 0;
lines[++tot] = node(y, head[x]), head[x] = tot;
}
template<typename T> inline T min(const T &a,const T &b){return a<b?a:b;}
template<typename T> inline T min(const T &a,const T &b,const T &c){return min(min(a,b),c);}
template<typename T> inline T max(const T &a,const T &b){return a>b?a:b;}
int n;
int w[1501];
int dp[1501][3];
//dp[i][0] 自己控制
//dp[i][1] 父亲控制
//dp[i][2] 儿子控制
void dfs(int now,int fa)
{
int v,minp = 999999999;
bool flag = true;//标记是否强制有一个儿子控制自己
dp[now][0] = w[now];
for(int p = head[now];p;p=lines[p].next)
{
v = lines[p].to;
if(v == fa)
continue;
dfs(v,now);
dp[now][0] += min(dp[v][2],dp[v][1],dp[v][0]);
dp[now][1] += min(dp[v][0],dp[v][2]);
if(dp[v][0] <= dp[v][2])
{
//此时可以直接选择这个儿子控制自己了
flag = false;
dp[now][2] += dp[v][0];
}
else
{
dp[now][2] += dp[v][2];
minp = min(minp,dp[v][0] - dp[v][2]);
}
}
if(flag)
dp[now][2] += minp;
}
int main()
{
read(n);
int m,u,x;
for(int i = 1;i<=n;++i)
{
read(u);
read(w[u]);
read(m);
while(m--)
{
read(x);
add(u,x);
add(x,u);//居然要双向边?!
}
}
dfs(1,0);
printf("%d",min(dp[1][0],dp[1][2]));
return 0;
}

[Luogu][P2458] [SDOI2006]保安站岗的更多相关文章

  1. Luogu P2458 [SDOI2006]保安站岗(树形dp)

    P2458 [SDOI2006]保安站岗 题意 题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下 ...

  2. Luogu P2458 [SDOI2006]保安站岗【树形Dp】

    题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下超市的所有通道呈一棵树的形状:某些通道之间可以互 ...

  3. C++ 洛谷 P2458 [SDOI2006]保安站岗 from_树形DP

    P2458 [SDOI2006]保安站岗 没学树形DP的,看一下. 题目大意:一棵树有N个节点,现在需要将所有节点都看守住,如果我们选择了节点i,那么节点i本身,节点i的父亲和儿子都会被看守住. 每个 ...

  4. [luogu 2458][SDOI2006]保安站岗

    题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下超市的所有通道呈一棵树的形状:某些通道之间可以互 ...

  5. P2458 [SDOI2006]保安站岗[树形dp]

    题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下超市的所有通道呈一棵树的形状:某些通道之间可以互 ...

  6. 洛谷 P2458 [SDOI2006]保安站岗

    题目传送门 解题思路: 树形DP 可知一个点被控制有且仅有一下三种情况: 1.被父亲节点上的保安控制 2.被儿子节点上的保安控制 3.被当前节点上的保安控制 我们设dp[0/1/2][u]表示u节点所 ...

  7. 洛谷【P2458】[SDOI2006]保安站岗 题解 树上DP

    题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下超市的所有通道呈一棵树的形状:某些通道之间可以互 ...

  8. [Luogu2458][SDOI2006]保安站岗

    题目描述 五一来临,某地下超市为了便于疏通和指挥密集的人员和车辆,以免造成超市内的混乱和拥挤,准备临时从外单位调用部分保安来维持交通秩序. 已知整个地下超市的所有通道呈一棵树的形状:某些通道之间可以互 ...

  9. [SDOI2006] 保安站岗

    题目链接 第一遍不知道为什么就爆零了…… 第二遍改了一下策略,思路没变,结果不知道为什么就 A 了??? 树形 DP 经典问题:选择最少点以覆盖树上所有点(边). 对于本题,设 dp[i][0/1][ ...

随机推荐

  1. Python开发:Python运算符

    运算符 1.算数运算: 运算符 描述 实例 + 加 - 两个对象相加 a + b 输出结果 30 - 减 - 得到负数或是一个数减去另一个数 a - b 输出结果 -10 * 乘 - 两个数相乘或是返 ...

  2. windows系统下hosts文件的改写(为了测试nginx内网的证书代理,需要做域名解析)

    1. win加R     C:\WINDOWS\system32\drivers\etc 2.打开hosts文件  加入一行  IP为客户机要访问的IP地址  域名也是在nginx中定义好的 3.ct ...

  3. Hibernate框架:org.hibernate.exception.SQLGrammarException: Cannot open connection at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.java92)

    报错信息 org.hibernate.exception.SQLGrammarException: Cannot open connection at org.hibernate.exception. ...

  4. 爬虫模拟cookie自动登录(人人网自动登录)

    什么是cookie? 在网站中,HTTP请求时无状态的,也就是说即使第一次和服务器连接后并且登录成功后,第二次请求服务器依然不能知道当前请求是谁,cookie的出现就是为了解决这个问题,第一次登陆后服 ...

  5. LinkedHashMap的用法

    1:LinkedHashMap的简介 Map 接口的哈希表和链接列表实现,具有可预知的迭代顺序.此实现与 HashMap 的不同之处在于,后者维护着一个运行于所有条目的双重链接列表.此链接列表定义了迭 ...

  6. 【PAT甲级】1024 Palindromic Number (25 分)

    题意: 输入两个正整数N和K(N<=1e10,k<=100),求K次内N和N的反置相加能否得到一个回文数,输出这个数和最小的操作次数. trick: 1e10的数字相加100次可能达到1e ...

  7. 「Luogu P1383 高级打字机」

    一道非常基础的可持久化数据结构题. 前置芝士 可持久化线段树:实现的方法主要是主席树. 具体做法 这个基本就是一个模板题了,记录一下每一个版本的字符串的长度,在修改的时候就只要在上一个版本后面加上一个 ...

  8. Linux 允许root用户远程登陆

    首先确保ssh服务已经安装: ps -e | grep ssh or service ssh start 如果没有安装则: apt-get install ssh 安装完之后 查看 /etc/ssh/ ...

  9. ES6 && ECMAScript2015 新特性

      ECMAScript 6(以下简称ES6)是JavaScript语言的下一代标准.因为当前版本的ES6是在2015年发布的,所以又称ECMAScript 2015. 也就是说,ES6就是ES201 ...

  10. C 如何判断编译器是否支持C90 C99?

    参考:<C Primer Plus>,Stephen Prata著,姜佑译. ANSI/ISO C标准 美国ANSI成立委员会X3J11,于89/90年,99年,11年,发布C标准:C89 ...