BZOJ原题链接

洛谷原题链接

该题有两种做法,树形\(DP\)和贪心。

先讲贪心。

先将所有点按深度从大到小排序,然后从大到小依次取出点,若已经被覆盖则跳过,否则就在它的祖父点建立消防站。

考虑如何判断该点是否被覆盖,设数组\(dis[x]\)表示点\(x\)到达离它最近的消防站的距离。

则在扫到一个点时,先用它父亲和祖父的\(dis\)来尝试更新该点的\(dis\),即尝试用父亲或祖父来覆盖该点,若没有被覆盖,则在祖父建立消防站,并更新祖父的父亲和祖父的祖父的\(dis\),这样就能同时将兄弟节点和儿子(孙子)节点覆盖点的情况记录下来。

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N = 1010;
int de[N], fa[N], po[N], dis[N];
inline int re()
{
int x = 0;
char c = getchar();
bool p = 0;
for (; c < '0' || c > '9'; c = getchar())
p |= c == '-';
for (; c >= '0' && c <= '9'; c = getchar())
x = x * 10 + c - '0';
return p ? -x : x;
}
inline int minn(int x, int y)
{
return x < y ? x : y;
}
bool comp(int x, int y)
{
return de[x] > de[y];
}
int main()
{
int i, x, y, z, n, s = 0;
n = re();
memset(dis, 60, sizeof(dis));
for (de[1] = po[1] = 1, i = 2; i <= n; i++)
{
scanf("%d", &fa[i]);
de[i] = de[fa[i]] + 1;
po[i] = i;
}
sort(po + 1, po + n + 1, comp);
for (i = 1; i <= n; i++)
{
x = po[i];
y = fa[x];
z = fa[y];
dis[x] = minn(dis[x], minn(dis[y] + 1, dis[z] + 2));
if (dis[x] > 2)
{
dis[z] = 0;
s++;
x = fa[z];
y = fa[x];
dis[x] = minn(dis[x], 1);
dis[y] = minn(dis[y], 2);
}
}
printf("%d", s);
return 0;
}

然后是另一种做法,树形\(DP\)。

设\(f[x][k]\)表示以\(x\)为根的子树,\(x\)点在\(k\)状态下建立的消防局总数。

  1. \(k = 0\)表示在\(x\)建立消防局。
  2. \(k = 1\)表示\(x\)至少有一个儿子建立消防局。
  3. \(k = 2\)表示\(x\)至少有一个孙子建立消防局。
  4. \(k = 3\)表示\(x\)的儿子节点全部被覆盖。
  5. \(k = 4\)表示\(x\)的孙子节点全部被覆盖。

前\(3\)种情况保证\(x\)被覆盖,后\(2\)种则不能保证。

设\(y,z\)均为\(x\)儿子。

则有状态转移方程:

\(\qquad\qquad f[x][0] = 1 + \sum\min\{ f[y][0\to 4] \}\)

\(\qquad\qquad f[x][1] = \min\{ f[y][0] + \sum\min\{ f[z][0\to 3]\ (z \neq y) \} \}\)

\(\qquad\qquad f[x][2] = \min\{ f[y][1] + \sum\min\{ f[z][0\to 2]\ (z \neq y) \} \}\)

\(\qquad\qquad f[x][3] = \sum\min\{ f[y][0\to 2] \}\)

\(\qquad\qquad f[x][4] = \sum\min\{ f[y][0\to 3] \}\)

显然对于状态\(0\to 4\),建立的消防局总数依次减小,所以可以优化转移方程:

\(\qquad\qquad f[x][0] = 1 + \sum f[y][4]\)

\(\qquad\qquad f[x][1] = f[x][4] + \min\{ f[y][0] - f[y][3] \}\)

\(\qquad\qquad f[x][2] = f[x][3] + \min\{ f[y][1] - f[y][2] \}\)

\(\qquad\qquad f[x][3] = \sum f[y][2]\)

\(\qquad\qquad f[x][4] = \sum f[y][3]\)

最后答案就是\(f[1][2]\)。

因为贪心的可拓展性较好(可以处理覆盖\(k\)距离的情况),且打起来简单,所以这里就不给出\(DP\)的代码了(其实是懒)。

BZOJ1217或洛谷2279 [HNOI2003]消防局的设立的更多相关文章

  1. 洛谷 2279 [HNOI2003]消防局的设立

    Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了 ...

  2. 洛谷 P2279 [HNOI2003]消防局的设立 (树形dp or 贪心)

    一看到这道题就知道是树形dp 之前做过类似的题,只不过保护的范围是1 所以简单很多. 这道题保护的范围是2,就复杂了很多. 我就开始列状态,然后发现竟然有5种 然后我就开始列方程. 但是我考虑的时候是 ...

  3. 洛谷P2279 [HNOI2003]消防局的设立

    题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了一个巨大的树状 ...

  4. 洛谷 P2279 [HNOI2003]消防局的设立

    题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了一个巨大的树状 ...

  5. 洛谷 P2279 [HNOI2003]消防局的设立 题解

    每日一题 day34 打卡 Analysis 这道题的正解本来是树形dp,但要设5个状态,太麻烦了.于是我就用贪心试图做出此题,没想到还真做出来了. 考虑当前深度最大的叶子结点,你肯定要有一个消防局去 ...

  6. Luogu 2279 [HNOI2003]消防局的设立 - 贪心

    Description 给定一棵树形图, 建若干个消防站, 消防站能够覆盖到距离不超过2的点, 求最少需要建几个消防站才能覆盖所有点 Solution 从深度最深的点开始, 在它的爷爷节点上建, 每建 ...

  7. luogu 2279 [HNOI2003]消防局的设立 树形dp

    就是细节多一些,思路都非常常规. Code: #include <bits/stdc++.h> #define N 1005 #define inf 1061109567 #define ...

  8. 【BZOJ1217】[HNOI2003]消防局的设立 树形DP

    [BZOJ1217][HNOI2003]消防局的设立 Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地, ...

  9. BZOJ1217: [HNOI2003]消防局的设立

    BZOJ1217: [HNOI2003]消防局的设立 Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地. 起初为了节约材料,人类只修建了n-1条道路来连接这些基地 ...

随机推荐

  1. python字典dict的增、删、改、查操作

    ## python字典dict的增.删.改.查操作dict = {'age': 18, 'name': 'jin', 'sex': 'male', }#增# dict['heigh'] = 185 # ...

  2. JQuery 基本知识

    一.简介 JQuery是继prototype之后又一个优秀的Javascript库.它是轻量级的js库 ,它兼容CSS3,还兼容各种浏览器(IE 6.0+, FF1.5+, Safari 2.0+,  ...

  3. Numpy知识(三)

    ndarray的花式索引. 正负数索引,正数就是从0开始的下标正向寻找,负数是-1开始的负向寻找. arr[[1,5,2,6],[0,3,1,2]]:拿取arr[1,0],arr[5,3],arr[2 ...

  4. Numpy知识(一)

    先了解ndarray(一个多维数组)的生成. 第一种生成方法就是np.random.randn(n,m),这回生成一个形状是n*m的ndarray.如下图. 第二种生成方法:传入一个列表listDem ...

  5. Python爬虫使用MD5加密的坑

    由于公司的业务需要,需要爬取很多的国外网站图片,然后兄弟我一路正则杀过去,总共匹配到658张链接,心里美滋滋开始写下载的代码.然后就有了这次坑的记录. 首先这是我查到的链接数量 然后爬虫跑完后,美滋滋 ...

  6. 在 MySQL 中创建一个中文数据库

    安装完 MySQL 后,要修改密码. step 1: SET PASSWORD = PASSWORD('your new password'); step 2: ALTER USER 'root'@' ...

  7. docker的理解

    作者:刘允鹏链接:https://www.zhihu.com/question/28300645/answer/67707287来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明 ...

  8. 58. Length of Last Word (String)

    Given a string s consists of upper/lower-case alphabets and empty space characters ' ', return the l ...

  9. ios系统微信浏览器、safari浏览器中h5页面上拉下滑导致悬浮层脱离窗口的解决方法

    一. 运行环境: iphone所有机型的qq浏览器,safari浏览器,微信内置浏览器(qq浏览器内核)等. 二. 异常现象: 1. 大幅度上下滑动h5页面,然后停止滑动,有时候会影响到页面滚动,如局 ...

  10. Gym - 101243F Vitamins(思维+并查集)

    题意 有三种药丸,白色W>红色R>蓝色B,给你m个约束条件,问你n个药丸的颜色,不能确定颜色输出‘?’ 题解 如果1<2<3,只要找到2就能确定1和3的颜色 如果2=4,只要确 ...