题目链接:传送门

题目大意:

  给定一棵树(N个基地N-1条边);

  用半径为2的消防局覆盖这N个基地,问最小的消防局数量。

  (树上距离为k的最小覆盖问题)

思路:

  每次贪心地找到不被覆盖的最深的一个节点,在它的祖父处放一个消防局。

  这个消防局所在位置即能将这个节点覆盖到的离它最远的点。

#include <bits/stdc++.h>

using namespace std;
const int MAX_N = 1e3 + ; int N;
vector <int> Edge[MAX_N];
int dep[MAX_N], fa[MAX_N];
bool vis[MAX_N]; int findlower()
{
int lower = , ind = ;
for (int i = ; i <= N; i++) {
if (!vis[i] && dep[i] > lower) {
lower = dep[i];
ind = i;
}
}
return ind;
} void update(int x)
{
x = fa[fa[x]];
for (int i = ; i < (int)Edge[x].size(); i++) {
int y = Edge[x][i];
vis[y] = true;
for (int j = ; j < (int)Edge[y].size(); j++) {
int z = Edge[y][j];
vis[z] = true;
}
}
} int solve()
{
memset(vis, false, sizeof vis);
int ans = ;
int cur = findlower();
while (cur) {
update(cur);
ans++;
cur = findlower();
}
return ans;
} void build(int x)//不妨把编号为1的点当作根节点
{
for (int i = ; i <= N; i++) {
dep[i] = -;
fa[i] = i;
}
dep[x] = ;
queue <int> Q;
Q.push(x);
while (!Q.empty()) {
int u = Q.front(); Q.pop();
for (int i = ; i < (int)Edge[u].size(); i++) {
int v = Edge[u][i];
if (dep[v] < ) {
dep[v] = dep[u] + ;
fa[v] = u;
Q.push(v);
}
}
}
} int main()
{
cin >> N;
for (int u = ; u <= N; u++) {
int v;
cin >> v;
Edge[u].push_back(v);
Edge[v].push_back(u);
}
int ans = N;
build();
ans = min(ans, solve());
cout << ans << endl;
return ;
}

  前面有点想多了,因为题目中给出的ai < i,所以不需要build直接拿题目中给的树来用就好了。。

  不过也无伤大雅。

  然后是膜大佬学到的代码:

  求树上距离为k的最小覆盖都可以这样做:

  (时间复杂度为O(N * k))

#include <bits/stdc++.h>

using namespace std;
const int MAX_N = 1e3 + ;
const int INF = 0x3f3f3f3f; struct Node{
int dep, ind;
Node(int d = , int i = ) : dep(d), ind(i) {}
bool operator < (const Node& x) const {
return dep > x.dep;
}
}nodes[MAX_N]; int fa[MAX_N], dis[MAX_N]; int main()
{
int N;
cin >> N;
nodes[] = Node(, );
fa[] = ;
dis[] = INF;
for (int i = ; i <= N; i++) {
scanf("%d", fa+i);
dis[i] = INF;
nodes[i].ind = i;
nodes[i].dep = nodes[fa[i]].dep + ;
}
sort(nodes+, nodes+N+);
int ans = ;
for (int i = ; i <= N; i++) {
int u = nodes[i].ind;
int v = fa[u];
int w = fa[v];
if (dis[u] > && dis[v] > && dis[w] > ) {
ans++;
dis[u] = ;
dis[v] = min(dis[v], );
dis[w] = min(dis[w], );
dis[fa[w]] = min(dis[fa[w]], );
dis[fa[fa[w]]] = min(dis[fa[fa[w]]], );
}
}
cout << ans << endl;
return ;
}

  

P2279 消防局的设立(贪心+dp)的更多相关文章

  1. P2279 消防局的设立 (树形DP or 贪心)

    (点击此处查看原题) 树形DP写法 看到这个题的要求,很容易相到这是一个树形DP的问题,但是dp数组应该如何设计并转移才是关键 dp[i][0]代表当前结点可以向上覆盖2层,自身一定被覆盖dp[i][ ...

  2. [luogu]P2279 [HNOI2003]消防局的设立[贪心]

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

  3. BZOJ 1217: [HNOI2003]消防局的设立( 贪心 )

    一个简单的贪心, 我们只要考虑2个消防局设立的距离为5时是最好的, 因为利用最充分. 就dfs一遍, 再对根处理一下就可以了. 这道题应该是SGU某道题的简化版...这道题距离只有2, 树型dp应该也 ...

  4. [HNOI2003]消防局的设立 (贪心)

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

  5. 【题解】P2279消防局的设立

    [题解][P2279 HNOI2003]消防局的设立 又是一道贪心. 随便指定一个点为根,可以知道在覆盖了一个节点的子树的情况下,消防站越高越好.那么我们就贪心吧.\(trick\)是按深度\(pus ...

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

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

  7. P2279 [HNOI2003]消防局的设立 贪心or树形dp

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

  8. [HNOI2003]消防局的设立 树形dp // 贪心

    https://www.luogu.org/problemnew/show/P2279 一开始就想到了贪心的方法,不过一直觉得不能证明. 贪心的考虑是在深度从深到浅遍历每个结点的过程中,对于每个没有覆 ...

  9. 洛谷P2279 消防局的设立 [HNOI2003] 贪心

    正解:贪心 解题报告: 传送门! 这题贪心得挺显然的,,,?居然能有蓝,,,是蓝题太水了嘛,,,? 简单说下,这题一看到就能想到,对最低的没被覆盖到的点给它的祖父建一个消防局 没了? 哦这题实现还挺有 ...

随机推荐

  1. npm run dev 报错 版本太低

    解决方案是: 先用命令: npm -v 查看下你的版本(我原来是 V3.1 不行) 然后用 cnpm install -g npm  更新版本 npm - v 变成最新的4.0.4 npm run d ...

  2. Linux第八周作业

    一 理解编译链接的过程和ELF可执行文件格式 这张图说明了可执行程序的产生 大致过程为 .c文件汇编成汇编代码.asm, 然后再汇编成目标码.o, 然后链接成可执行文件a.out, 这时可执行文件就可 ...

  3. Ubuntu16.10下mysql5.7的安装及远程访问配置

    如何安装mysql 1.sudo apt-get update,如果很慢或者失败,需要在软件和更新中选择最佳服务器,勾选所有互联网下载选项及去掉其他软件所有勾选项 2.sudo apt-get upg ...

  4. java this的用法

    this 含义:代表当前对象 用法: 用于返回对象的引用 示例代码 public class Test { public Test f() { return this;//获取当前对象的引用 } pu ...

  5. JavaMail发送邮件、带附件邮件(完整版)

    工程目录如下: 1.准备javaMail需要的两个Jar包:mail.jar.activation.jar,然后add to build path 2.QQ邮箱开启SMTP服务,开启后,它会给你一串授 ...

  6. 解决WDCP3环境gbk网站编码程序乱码问题

    因为默认WDCP V3版本环境编码格式是UTF-8版本,如果我们程序采用的是GBK编码肯定都会有乱码问题. 我们到WDCP后台,"网站管理"-"PHP设置",看 ...

  7. Xcode清理存储空间

    文章来自 枣泥布丁 http://www.cocoachina.com/ios/20170711/19814.html 请针对性的选择删除 移除 Xcode 运行安装 APP 产生的缓存文件(Deri ...

  8. vector容器的注意事项

    1.容器是指对象的集合,每一个元素都是一个对象,并且对象的类型相同.可以使用索引去访问容器中的对象. 2.由于容器中存放的是对象,所以引用无法成为vector的成员. 3.容器的初始化,与string ...

  9. java动手动脑3

    2016-10-152016-10-15一.编写一个方法,使用以上算法生成指定数目(比如1000个)的随机整数. 生成50个1到10的随机整数. value=a+(int)(Math.Random() ...

  10. Centos7安装配置MySQL5.7

    一:安装前准备: 1.1检查linux版本:cat /etc/system-release CentOS Linux release 7.6.1810 (Core) 1.2查看系统是否安装MySQL: ...