不能走路(walk)
【题目背景】
小G 同学总是在树上走路。小S 看不下去了,决定阻止小G 同学。
【题目描述】
有一棵 n 个点的树,树上有 m 条路径,每条路径为 x[i]到y[i] 的树上最短路径(不经过相同的边),小 S 想要摧毁树上的 K 个点, 使得每一条路径上都有至少一个点被摧毁。
你只需要求出最小的K,而不用输出摧毁了哪些点。
| 数据点编号 | n<= | m<= |
|---|---|---|
| 1 | 15 | 15 |
| 2 | 100 | 15 |
| 3 | 100 | 15 |
| 4 | 100 | 100 |
| 5 | 1000 | 1000 |
| 6 | 5000 | 5000 |
| 7 | 5000 | 5000 |
| 8 | 100000 | 100000 |
| 9 | 100000 | 100000 |
| 10 | 100000 | 100000 |
【样例输入】
5 2
1 2
2 3
2 4
2 5
1 5
3 4
【样例输出】
1
【提示】
如果需要用到较大的栈空间。
可以在编译命令加上-Wl,--stack=2333333333。评测时开栈。
题解
受洛谷P1967 货车运输启发,这题很像LCA。
于是我们就尝试切LCA。
首先,不难发现阻断LCA是可行的,那么是最优的吗?
对于一颗子树内的点,如果从该子树的点要延伸出子树外,那必须经过最顶端的点。
如图,对于一个点,若要与外面有联系,那必定要经过\(lca(a,b)\)。
那两点的联系在子树内呢?如果是\(c\to d\),那显然是无关的,那\(e\to b\)呢?我们发现最优应该切\(lca(e, b)\)。

那不是就WA了吗?不。我们发现:如果我们先切\(lca(e, b)\),到\(a\to b\)是特判一下即可。
由于数据很水,判是否联通用暴力爬就可以了。
下面上代码,刚学树剖,于是就打了个树剖。
#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn = 100005;
struct Edge
{
int to, nxt;
} e[maxn<<1];
int first[maxn];
int nowm;
inline void add_edge(int from, int to)
{
e[++nowm].nxt = first[from];
e[nowm].to = to;
first[from] = nowm;
e[++nowm].nxt = first[to];
e[nowm].to = from;
first[to] = nowm;
}
int fa[maxn], son[maxn], dep[maxn], val[maxn];
inline void dfs1(int x)//找重链,标记深度与父亲
{
dep[x] = dep[fa[x]] + 1;
val[x] = 1;
for(int i = first[x], dd; i; i = e[i].nxt)
{
dd = e[i].to;
if(dd == fa[x])
continue;
fa[dd] = x;
dfs1(dd);
val[x] += val[dd];
if(!son[x] || val[dd] > val[son[x]])
son[x] = dd;
}
}
int top[maxn];
inline void dfs2(int x, int ff)//寻找重链的顶端
{
top[x] = ff;
if(son[x])
dfs2(son[x], ff);
for(int i = first[x], dd; i; i = e[i].nxt)
{
dd = e[i].to;
if(dd != fa[x] && dd != son[x])
dfs2(dd, dd);
}
}
inline void init(int s)
{
dfs1(s);
dfs2(s, s);
}
inline int lca(int a, int b)
{
while(top[a] != top[b])
{
if(dep[top[a]] >= dep[top[b]])
a = fa[top[a]];
else
b = fa[top[b]];
}
return dep[a] < dep[b] ? a : b;
}
struct sxd
{
int a, b, t;
inline bool operator < (const sxd& other) const
{
return dep[t] > dep[other.t];
}
} ask[maxn];
bool biao[maxn];
inline bool pan(int a, int b)
{
for(; a != fa[b]; a = fa[a])
if(biao[a])
return true;
return false;
}
int main()
{
freopen("walk.in", "r", stdin);
freopen("walk.out", "w", stdout);
int n, m;
scanf("%d%d", &n, &m);
for(int i = 1, f, t; i < n; ++i)
{
scanf("%d%d", &f, &t);
add_edge(f, t);
}
init(1);
for(int i = 1; i <= m; ++i)
{
scanf("%d%d", &ask[i].a, &ask[i].b);
ask[i].t = lca(ask[i].a, ask[i].b);//树剖lca模板
}
sort(ask+1, ask+m+1);//将提问排序,从下往上阻断
int ans = 0;
for(int i = 1; i <= m; ++i)//暴力往上爬……(竟不会T,还是最优解)
{
if(pan(ask[i].a, ask[i].t) || pan(ask[i].b, ask[i].t))
continue;
ans++;
biao[ask[i].t] = true;
}
printf("%d\n", ans);
return 0;
}
不能走路(walk)的更多相关文章
- 学C#之设计模式系列笔记(1)策略模式
一.借鉴说明 1.<Head First Design Patterns>(中文名<深入浅出设计模式>) 2.维基百科,策略模式,https://zh.wikipedia.or ...
- Quick StateMachine状态机
状态机quick中是一个亮点,假设我们做一款RPG游戏,一个角色通常会拥有idle,attack,walk.run,death这些状态,假设游戏角色的状态採用分支条件推断的话.会造成很庞大而难以维护. ...
- Openjudge-计算概论(A)-骑车与走路
描述: 在北大校园里,没有自行车,上课办事会很不方便.但实际上,并非去办任何事情都是骑车快,因为骑车总要找车.开锁.停车.锁车等,这要耽误一些时间.假设找到自行车,开锁并车上自行车的时间为27秒;停车 ...
- hdu_A Walk Through the Forest ——迪杰特斯拉+dfs
A Walk Through the Forest Time Limit : 2000/1000ms (Java/Other) Memory Limit : 65536/32768K (Java/ ...
- 计算概论(A)/基础编程练习2(8题)/4:骑车与走路
#include<stdio.h> int main() { // 待处理的数据数量n ; scanf("%d", &n); float meters[n]; ...
- Spoken English Practice(And I can't walk. Should i just stay home and pout about it?)
绿色:连读: 红色:略读: 蓝色:浊化: 橙色:弱读 下划线_为浊化 口语蜕变(2017/7/12) ...
- 【ACM】bailian_2703_骑车与走路_201307301543
2703:骑车与走路总时间限制: 1000ms 内存限制: 65536kB 描述 在北大校园里,没有自行车,上课办事会很不方便.但实际上,并非去办任何事情都是骑车快,因为骑车总要找车.开锁.停车.锁车 ...
- OpenJudge计算概论-骑车与走路
/*============================================================ 骑车与走路 总时间限制: 1000ms 内存限制: 65536kB 描述 ...
- python os.walk()
os.walk()返回三个参数:os.walk(dirpath,dirnames,filenames) for dirpath,dirnames,filenames in os.walk(): 返回d ...
随机推荐
- 【06月05日】A股滚动市净率PB历史新低排名
2010年01月01日 到 2019年06月05日 之间,滚动市净率历史新低排名. 上市三年以上的公司,2019年06月05日市净率在30以下的公司. 来源:A股滚动市净率(PB)历史新低排名. 1 ...
- activiti学习7:spring和activiti进行整合
目录 activiti学习7:spring和activiti进行整合 一.整合原理 二.整合步骤 2.1 新建一个maven工程并导入相关依赖 2.2 创建spring配置文件 三.测试 activi ...
- Java8实战及自己的总结
java8 介绍 java8是2014年3月份,由Oracle发布的一个版本,又称之为jdk1.8,是现再我们在学习和工作中用的最多的一个版本. 在jdk1.8中,java8以添加非常多的新特性, ...
- 什么是 Shell?
简单来说“Shell编程就是对一堆Linux命令的逻辑化处理”. Shell 编程的 Hello World 学习任何一门编程语言第一件事就是输出HelloWord了!下面我会从新建文件到shell代 ...
- requests获取响应时间(elapsed)与超时(timeout)、小数四舍五入
前言 requests发请求时,接口的响应时间,也是我们需要关注的一个点,如果响应时间太长,也是不合理的.如果服务端没及时响应,也不能一直等着,可以设置一个timeout超时的时间 elapsed官方 ...
- Spring Boot 主从读写分离
自己封装了一个读写分离的 Starter,可以配置任意多个数据源,使用 Hikari 连接池(暂不支持其他连接池). GitHub:rw-separate-spring-boot-starter 代码 ...
- framework7 下拉刷新、无限滚动
下拉刷新: html: <div class="page-content ptr-content"> <%--下拉刷新图标--%> <div clas ...
- Spring Security实现OAuth2.0授权服务 - 基础版
一.OAuth2.0协议 1.OAuth2.0概述 OAuth2.0是一个关于授权的开放网络协议. 该协议在第三方应用与服务提供平台之间设置了一个授权层.第三方应用需要服务资源时,并不是直接使用用户帐 ...
- jquery实现简单定时轮播图
JS $(document).ready(function(){ var index = 0; //刚开始设置一个index,index为图片的索引值 $(".pictureDemo img ...
- 【方法论】5WHY分析法
一.概述 所谓“5WHY”分析法,又称“5问法”,就是连续反复使用5次“为什么”方式自问,以打破砂锅问到底方式寻找问题的根本原因的方法.“5WHY”不限定必须或只做5次为什么的提问,以找到问题根因为准 ...