题目链接:传送门

题目大意:

  给定一棵树(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. javascript void函数

    <a href="javascript:doTest2();void(0);">here</a> 但这儿的void(0)究竟是何含义呢? Javascrip ...

  2. C/S与B/S架构对比

    概述 在这个信息急剧膨胀的社会,我们不得不说人类正进入一个崭新的时代,那就是信息时代.信息时代的一个主要而显著的特征就是计算机网络的应用.计算机网络从最初的集中式计算,经过了Client/Server ...

  3. python 调用zabbix api实现查询主机信息,输出所有主机ip

    之前发现搜索出来的主机调用zabbix api信息都不是那么明确,后来通过zabbix官方文档,查到想要的api信息,随后写一篇自己这次项目中用到的api. #!/usr/bin/env python ...

  4. zookeeper集群环境搭建(使用kafka的zookeeper搭建zk集群)

    ---恢复内容开始--- 使用kafka的zookeeper来搞集群的话和单纯用zk的其实差不了多少. 0.说在前头,搭建kafka集群之前请把每个服务器的jdk搞起来. 1.安装kafka wget ...

  5. loadrunner请求json数据参数化问题

    http://blog.sina.com.cn/s/blog_62079f620102vvx3.html

  6. 7 Serial Configuration 理解 (一)

    reference :  ug470- 7 series config.pdf 7系列器件有5种配置接口,每种配置接口对应一种或者多种配置模式和总线位宽.配置时序相对于引脚的CCLK,即使在内部产生C ...

  7. Centos7创建用户su登录后显示为 bash-4.1$

    useradd name [root@localhost data]# su name bash-4.2$ [root@localhost ~]# cp -a /etc/skel/. /home/na ...

  8. javascript动态加载js文件主流浏览器兼容版

    一.代码示例: <html> <head> <meta http-equiv="Content-Type" content="text/ht ...

  9. usort 函数

    function getNameFromNumber($num){ // Used to figure out what the Excel column name would be for a gi ...

  10. js闭包 选择器 面向对象 事件 操作页面

    闭包js函数的嵌套定义,定义在内部的函数 就称之为闭包为什么使用闭包: 1.一个函数要使用另一个函数的局部变量 2.闭包会持久化包裹自身的函数的局部变量 3.解决循环绑定 function outer ...