P2279 消防局的设立(贪心+dp)
题目链接:传送门
题目大意:
给定一棵树(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)的更多相关文章
- P2279 消防局的设立 (树形DP or 贪心)
(点击此处查看原题) 树形DP写法 看到这个题的要求,很容易相到这是一个树形DP的问题,但是dp数组应该如何设计并转移才是关键 dp[i][0]代表当前结点可以向上覆盖2层,自身一定被覆盖dp[i][ ...
- [luogu]P2279 [HNOI2003]消防局的设立[贪心]
[luogu]P2279 [HNOI2003]消防局的设立 题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两 ...
- BZOJ 1217: [HNOI2003]消防局的设立( 贪心 )
一个简单的贪心, 我们只要考虑2个消防局设立的距离为5时是最好的, 因为利用最充分. 就dfs一遍, 再对根处理一下就可以了. 这道题应该是SGU某道题的简化版...这道题距离只有2, 树型dp应该也 ...
- [HNOI2003]消防局的设立 (贪心)
[HNOI2003]消防局的设立 题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达, ...
- 【题解】P2279消防局的设立
[题解][P2279 HNOI2003]消防局的设立 又是一道贪心. 随便指定一个点为根,可以知道在覆盖了一个节点的子树的情况下,消防站越高越好.那么我们就贪心吧.\(trick\)是按深度\(pus ...
- 【BZOJ1217】[HNOI2003]消防局的设立 树形DP
[BZOJ1217][HNOI2003]消防局的设立 Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地, ...
- P2279 [HNOI2003]消防局的设立 贪心or树形dp
题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了一个巨大的树状 ...
- [HNOI2003]消防局的设立 树形dp // 贪心
https://www.luogu.org/problemnew/show/P2279 一开始就想到了贪心的方法,不过一直觉得不能证明. 贪心的考虑是在深度从深到浅遍历每个结点的过程中,对于每个没有覆 ...
- 洛谷P2279 消防局的设立 [HNOI2003] 贪心
正解:贪心 解题报告: 传送门! 这题贪心得挺显然的,,,?居然能有蓝,,,是蓝题太水了嘛,,,? 简单说下,这题一看到就能想到,对最低的没被覆盖到的点给它的祖父建一个消防局 没了? 哦这题实现还挺有 ...
随机推荐
- JSP调试技巧
我先谈谈: 我的经验就是多装几个服务器,这个查不出错误,用另一个,这个方法很好用. ---------------------------------------------------------- ...
- js重写系统的弹框
//调用系统的弹框,不显示地址 window.alert = function(name){ var iframe = document.createElement("IFRAME& ...
- linux basic test
Linux 1◆ 提供连接 2◆ connection baidu.com 3◆ vm tools install Reboot
- H5 PWA技术以及小demo
H5 PWA技术 1.原生app优缺点 a.体验好.下载到手机上入口方便 b.开发成本高(ios和安卓) c.软件上线需要审核 d.版本更新需要将新版本上传到不同的应用商店 e.使用前需下载 2.we ...
- UVa LA 3213 - Ancient Cipher 水题 难度: 0
题目 https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_pr ...
- POJ 1837 Balance 水题, DP 难度:0
题目 http://poj.org/problem?id=1837 题意 单组数据,有一根杠杆,有R个钩子,其位置hi为整数且属于[-15,15],有C个重物,其质量wi为整数且属于[1,25],重物 ...
- Factory,工厂设计模式,C++描述
body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...
- vue组件的使用和事件传递
子组件与父组件的事件传递具体实现如下: 子组件: <template> <section class="xftz-data-list"> <div c ...
- Java 集成开发环境的介绍及下载
集成开发环境(integrated development environment,JDE) 之前成功运行了Java小程序是经历了先在笔记本中编写源代码,然后通过命令行运行打开javac编译源文件, ...
- 3.6 C++继承机制下的构造函数
参考:http://www.weixueyuan.net/view/6363.html 总结: 在codingbook类中新增了一个language成员变量,为此必须重新设计新的构造函数.在本例中bo ...