「HNOI2003」消防局的设立
题目
题目描述
输入格式
输出格式
输出文件仅有一个正整数,表示至少要设立多少个消防局才有能力及时扑灭任何基地发生的火灾。
样例
样例输入
6
1
2
3
4
5
样例输出
2
数据范围与提示
题解
做题经历
刚开始看,感觉这道题是一道较简单的树 $dp$ ,于是我想练练我贫穷的 $dp$ 技巧......结果花了差不多一个半小时得到了 $10pts$ ,然后在 $zxy$ 大佬的帮助下我当场 $AC$。
首先我的 $dp$ 定义:$dp[u][0|1]$:第 $u$ 个点建立/不建立消防站
于是我连 $dp$ 式都写不出来.....
正解
法Ⅰ:贪心的 $zxy$ 大佬
每访问到一个 $u$ 节点时,找一个二元组 $(minn,maxx)$ 表示在 $u$ 这棵子树中,建了消防站的点距离 $u$ 的距离最小值与在 $u$ 这棵子树上,还未被覆盖的点距离 $u$ 的距离最大值。
先贴一张图片,方便说明:
(这张图只是方便说明,可能与此题无关)
我们先来考虑这个二元组 $(minn,maxx)$ 。
首先我们需要保证的是,每个 $u$ 节点的儿子 $v$ ,对于以 $v$ 为根的子树都是已经被处理好的,即这个二元组 $(minn,maxx)$ 中的两个元素不能出现在同一棵子树中。
那么,这个已经被建立了消防站的点距离还未建立消防站的店的距离就是 $minn+maxx$。
假设我们访问到上图中的 $u=4$ 号节点,再假设距离其最近的且是其子树中的消防站是 $5$ ,而还未被覆盖的点是 $7$
那么可以计算数数得到 $minn=1,maxx=2$
那么节点 $5$ 与节点 $7$ 的距离就是 $dis=minn+maxx=1+2=3$
通过题目,我们知道一个消防站的染色范围是 $2$ ,也就是说如果这个 $dis≤2$ ,则说明这一整棵子树都已经被覆盖了(此处细想)
而如果 $dis>2$ ,则说明这个消防站无法覆盖完这棵树,那么就可能需要这个节点 $u$ 来建消防站来染色
为什么是可能呢?
我们来分析这样一种情况:$minn=4,maxx=1$
那么我们是不是必须在 $u$ 节点建消防站呢?
答案是不必须的,而且为了满足答案最优,我们还不能在此建立消防站。
为什么?
因为在 $maxx=1$ 时,说明如果我们在 $u$ 的父亲建立消防站,也是一样可以覆盖到这个尚未被覆盖的点。
而且可以保证,在 $u$ 的父亲建消防站是一定比在 $u$ 建消防站优的。
为什么?
还是同一张图:
如果我们选择在节点 $4$ 建立消防站,而尚未被覆盖的点是 $6$ 。
想一想,在 $2$ 建立消防站是否一定比在 $4$ 更优?
我们来看一看,如果在 $4$ 建立消防站,可以覆盖的点就只有 $6、4$ 和 $2、1$(假设 $5、7$ 是已经被覆盖了的)
但是如果我们在 $2$ 建立消防站,可以覆盖的点就有 $3、1、2、4、6$
所以,在其他的节点都已被处理好的前提下,将消防站尽量往高处建是最优的。
但是什么时候不得不建呢?
那么就是当$maxx+minn>2且maxx=2$时,如果我们再往上回溯,那么 $maxx$ 是会大于 $2$ 的,就是说在其父亲节点建消防站时,是已经够不到那个离得最远的点的。
时间复杂度$O(能过)$,代码如下:
#include<bits/stdc++.h>
using namespace std;
template<class T>inline void qread(T& x){
char c;bool f=false;x=0;
while((c=getchar())<'0'||'9'<c)if(c=='-')f=true;
for(x=(c^48);'0'<=(c=getchar())&&c<='9';x=(x<<1)+(x<<3)+(c^48));
if(f)x=-x;
}
template<class T,class... Args>inline void qread(T& x,Args&... args){qread(x),qread(args...);}
inline int rqread(){
char c;bool f=false;int x=0;
while((c=getchar())<'0'||'9'<c)if(c=='-')f=true;
for(x=(c^48);'0'<=(c=getchar())&&c<='9';x=(x<<1)+(x<<3)+(c^48));
return f?-x:x;
}
template<class T>inline T Max(const T x,const T y){return x>y?x:y;}
template<class T>inline T Min(const T x,const T y){return x<y?x:y;}
template<class T>inline T fab(const T x){return x>0?x:-x;}
const int MAXN=1000;
const int INF=0x3f3f3f3f;
struct edge{
int to,nxt;
edge(){}
edge(const int T,const int N):to(T),nxt(N){}
}e[(MAXN<<1)+5];
int tail[MAXN+5],edgeind;
inline void add_edge(const int u,const int v){
e[++edgeind]=edge(v,tail[u]);tail[u]=edgeind;
e[++edgeind]=edge(u,tail[v]);tail[v]=edgeind;
}
int N,dep[MAXN+5],tot;
struct node{
int minn,maxx;
node(){}
node(const int N,const int X):minn(N),maxx(X){}
};//定义的二元组
void buildtre(const int u,const int pre){
dep[u]=dep[pre]+1;
for(int i=tail[u],v;i;i=e[i].nxt)if((v=e[i].to)!=pre)
buildtre(v,u);
}
node dfs(const int u,const int pre){
node t,ret=node(INF,0);
for(int i=tail[u],v;i;i=e[i].nxt)if((v=e[i].to)!=pre){
t=dfs(v,u);
ret.minn=Min(ret.minn,t.minn);
ret.maxx=Max(ret.maxx,t.maxx);
}
if(ret.minn+ret.maxx<=2)return node(ret.minn+1,0);
if(ret.maxx==2)return ++tot,node(1,0);//除非不得不建,否则尽量往上走
return node(ret.minn+1,ret.maxx+1);
}
signed main(){
qread(N);
for(int i=2;i<=N;++i)add_edge(i,rqread());
buildtre(1,0);
node t=dfs(1,0);
if(t.maxx)++tot;//特殊处理根节点未覆盖的情况
printf("%d\n",tot);
return 0;
}
法Ⅱ:$lj$ 大佬的树 $dp$
我们分析这个消防站对于当前点的影响.
先来看这样一棵树:
假设我们已经访问到节点 $1$。
我们先考虑在节点 $2$ 建立消防站,那么它可以覆盖到 $10、1、7$ 和 $3、4$
如果在节点 $3$ 建立消防站,可覆盖 $1、2、4、5$
不难发现,如果我们建立的消防站的深度不同,其覆盖的节点构成(我在前文用‘和’分开了不同的构成,自己区分)(区分节点)也是不同的
那么,这个 $dp$ 的定义是不是会和建立消防站的深度或者距离有关。
那么定义二维状态:
$dp[u][x]$:在节点 $u$ 的子树中,所建立的消防站离它的距离为 $x$ 时,所建立的最少的消防站。
那么又有一个问题,这个 $x$ 的取值为多少呢?
先将一棵树拆成一条单链来看
- 当 $x=0$ 时,毫无疑问,是在 $1$ 建立的消防站
- 当 $x=1$ 时,在 $2$ 建消防站,那么 $1、3、4$都是被覆盖的
- 当 $x=2$ 时,同上
- 当 $x=3$ 时,在 $4$ 建消防站。那么有一个问题,节点 $1$ 好像没有被覆盖?那么 $1$ 怎么覆盖呢?只有可能是其爷爷或者其父亲。
- 当 $x=4$ 时,在 $5$ 建消防站,同样有个问题, $1、2$ 没有被覆盖,那么就需要 $1$ 的父亲来覆盖了
- 当 $x=5$ 时,在 $6$ 建消防站。这下问题就变大了, $1、2、3$ 都没有被覆盖,无论是在爷爷还是父亲节点都无法覆盖 $3$ 了,这下就必须在 $1$ 建立节点(尽量将消防站建高)才能覆盖 $3$ ,那么与 $x=0$ 重复
我们发现,当 $x=5$ 时,状态已经重复,说明 $x$ 最大的取值就为 $5$。
现在开始考虑状转:
- 当 $x=0$ 时,$dp[u][0]=\sum dp[v][4]$
- 当 $x=1$ 时,$dp[u][1]=\sum dp[v][3]+min\{dp[v][0]-dp[v][3]\}$
- 当 $x=2$ 时,$dp[u][2]=\sum dp[v][1]$
- 当 $x=3$ 时,$dp[u][3]=\sum dp[v][2]$
- 当 $x=4$ 时,$dp[u][4]=\sum dp[v][3]$
- 当 $x=5$ 时,不用管 $dp[u][5]$,因为上面的状态已经不可能用到它了(仔细观察所调用到的 $dp[v][x]$ ,都没有发现 $dp[v][5]$ 的出现)
再加以修饰,又是一份 $AC$ 代码:摘自 $trymyedge(lj)$ 大佬:
#include <bits/stdc++.h>
#define mz 1000000007
using namespace std; vector<int> vec[10005];
int dp[10005][5]; void dfs(int x, int fa) {
dp[x][0] = 1;
int minn = 9999;
for (int i = 0; i < vec[x].size(); i++) {
if (vec[x][i] != fa) {
dfs(vec[x][i], x);
dp[x][0] += dp[vec[x][i]][4];
dp[x][1] += dp[vec[x][i]][3];
minn = min(minn, dp[vec[x][i]][0] - dp[vec[x][i]][3]);
dp[x][2] += dp[vec[x][i]][1];
dp[x][3] += dp[vec[x][i]][2];
dp[x][4] += dp[vec[x][i]][3];
}
}
dp[x][1] += minn;
if (dp[x][2] == 0)
dp[x][2] = 9999;
for (int i = 1; i <= 4; i++) dp[x][i] = min(dp[x][i - 1], dp[x][i]);
} int main() {
int n, x;
scanf("%d", &n);
for (int i = 2; i <= n; i++) {
scanf("%d", &x);
vec[x].push_back(i);
vec[i].push_back(x);
}
dfs(1, 0);
printf("%d\n", min(dp[1][0], min(dp[1][1], dp[1][2])));
return 0;
}
「HNOI2003」消防局的设立的更多相关文章
- BZOJ 1217: [HNOI2003]消防局的设立( 贪心 )
一个简单的贪心, 我们只要考虑2个消防局设立的距离为5时是最好的, 因为利用最充分. 就dfs一遍, 再对根处理一下就可以了. 这道题应该是SGU某道题的简化版...这道题距离只有2, 树型dp应该也 ...
- P2279 [HNOI2003]消防局的设立
P2279 [HNOI2003]消防局的设立考场上想出了贪心策略,但是处理细节时有点问题,gg了.从(当前深度最大的节点)叶子节点往上跳k个,在这里设消防局,并从消防局遍历k个距离,标记上. #inc ...
- 【BZOJ1217】[HNOI2003]消防局的设立 树形DP
[BZOJ1217][HNOI2003]消防局的设立 Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地, ...
- [HNOI2003]消防局的设立 (贪心)
[HNOI2003]消防局的设立 题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达, ...
- BZOJ1217: [HNOI2003]消防局的设立
BZOJ1217: [HNOI2003]消防局的设立 Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地. 起初为了节约材料,人类只修建了n-1条道路来连接这些基地 ...
- [luogu]P2279 [HNOI2003]消防局的设立[贪心]
[luogu]P2279 [HNOI2003]消防局的设立 题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两 ...
- 【洛谷P2279】[HNOI2003]消防局的设立
消防局的设立 题目链接 贪心:每次取出深度最大的节点,若没有被覆盖到,要想覆盖它, 最优的做法显然是将它的爷爷设为消防局 (因为该节点深度为最大,选兄弟.父亲所覆盖的节点,选了爷爷后都能够覆盖) 用优 ...
- 「bzoj1925」「Sdoi2010」地精部落 (计数型dp)
「bzoj1925」「Sdoi2010」地精部落---------------------------------------------------------------------------- ...
- LOJ2722 「NOI2018」情报中心
「NOI2018」情报中心 题目描述 C 国和D 国近年来战火纷飞. 最近,C 国成功地渗透进入了D 国的一个城市.这个城市可以抽象成一张有$n$ 个节点,节点之间由$n - 1$ 条双向的边连接的无 ...
随机推荐
- python3+requests+BeautifulSoup+mysql爬取豆瓣电影top250
基础页面:https://movie.douban.com/top250 代码: from time import sleep from requests import get from bs4 im ...
- @media screen 自适应笔记
在css中使用@media screen 通过检索宽度 对应改变html中class的css属性. 1280分辨率以上(大于1200px) @media screen and (min-width:1 ...
- pandas read excel or csv
import pandas as pd """pandas doc:df.dtypes 查看数据每column 数据类型 id int64x0 float64df.rei ...
- 老段带你学鸟哥Linux视频教程 包含基础班+提高班
老段带你学鸟哥Linux视频教程 包含基础班+提高班,附带pdf文档. 目录结构如下: 目录:/-老段带你学鸟哥Linux视频教程 [.9G] ┣━━老段带你学鸟哥-服务器篇 [1009.4M] ┃ ...
- 3_6 环状序列(UVa1584)
长度为n的环状串有n种表示法,分别为某个位置开始顺时针得到.例如,图中的环状串有10种表示: CGAGTCAGCT,GAGTCAGCTC,AGTCAGCTCG等.在这些表示法中,字典序最小的称为“最小 ...
- jmh 微基准测试
选择依据:对某段代码的性能测试. 1.运行方法 mvn clean install java -jar target/benchmarks.jar JMHSample_02 -f 1 2.maven ...
- CSP-201609-3 炉石传说
问题描述 <炉石传说:魔兽英雄传>(Hearthstone: Heroes of Warcraft,简称炉石传说)是暴雪娱乐开发的一款集换式卡牌游戏(如下图所示).游戏在一个战斗棋盘上进行 ...
- Jmeter进行分布式性能测试
由于Jmeter本身的瓶颈,当需要模拟数以千计的并发用户时,使用单台机器模拟所有的并发用户就有些力不从心,甚至还会引起JAVA内存溢出的错误.要解决这个问题,可以使用分布式测试,运行多台机器运行所谓的 ...
- 小程序云函数调用http或https请求外部数据
参考网址 https://blog.csdn.net/qiushi_1990/article/details/101220920 小程序云函数调用http或https请求外部数据 原创编程小石头 发布 ...
- 整个DIV 块垂直居中
<!-- 垂直居中一定要有确定大小的 父容器(根) html,body 一般100% --> <!DOCTYPE html> <html lang="en&qu ...