题意:

有一棵树,树上有许多人,他们要聚会,找一个点使得所有人到这个点的距离的最大值最小。

题解:

首先,以一个有人的点为根,求一个生成树,删掉所有没有人的子树,保证所有的悬挂点(只连接一条边的点)都是有人的节点,以保证后面求出的直径的两端是两个有人节点。为什么非得以有人的节点为根呢?因为如果找了一个没人的节点当根,而这个根又恰好是一个悬挂点,那么可以想象,这个悬挂点可能成为直径的端点。

其次,求树的直径,就是树上两点的最长距离,这个直径的两个端点必然都是悬挂点。

直径/2向上取整就是半径,也就是题目所求距离。

求直径可跑两遍dfs,第一遍从任意点开始,找出与此点距离最大点,第二遍从这个点开始,找出与这个点距离最大点。

赛时想的是树的重心,wa了,想像一棵树上半部分是一条长链,下半部分是密密麻麻的儿子接在长链一端。然后又想了拓扑求直径,T了。

但是发现拓扑排序的轮数其实就是树的半径,不知道这个结论有什么用。

#include<iostream>
#include<vector>
#define INF 0x3f3f3f3f
#include<queue>
#include<cstring>
#include<set>
using namespace std;
vector<int> lnk[];
vector<int> sclnk[];//生成树
int n; int scn;//生成树的节点数
bool ren[];
bool dfs(int fa,int u){
bool ok=ren[u];
for(int i=;i<lnk[u].size();i++){
int v=lnk[u][i];
if(v==fa)continue;
bool tr=dfs(u,v);
if(tr){
sclnk[u].push_back(v);
sclnk[v].push_back(u);
scn++;
}
ok=ok||tr;
}
return ok;
}
int kg;
void prt(int fa,int u){
//把生成树打印出来
for(int i=;i<=kg;i++)printf(" ");
printf("%d\n",u);
kg++;
for(int i=;i<sclnk[u].size();i++){
int v=sclnk[u][i];
if(v==fa)continue;
prt(u,v);
}
kg--;
} bool vis[];
int size[]; int dis[];
struct P{
int n,dis;
P(){}
P(int n1,int dis1){
n=n1;dis=dis1;
}
friend bool operator >(const P &a,const P &b){
return a.dis>b.dis;
}
friend bool operator <(const P &a,const P &b){
return a.dis<b.dis;
}
}; void ddfs(int st) {
vis[st] = ;
for(int i = ;i<sclnk[st].size();i++) {
int to =sclnk[st][i];
if(!vis[to]) {
dis[to] = dis[st] +;
ddfs(to);
}
}
} int main(){
//把一个有人的点揪出来当树根
//删掉无人的子树
//找出树的直径
int k;
scanf("%d %d",&n,&k);
for(int i=;i<n;i++){
int a,b;
scanf("%d %d",&a,&b);
lnk[a].push_back(b);
lnk[b].push_back(a);
}
int r1;
for(int i=;i<=k;i++){
int q;
scanf("%d",&q);
r1=q;
ren[q]=;
}
dfs(-,r1);
// prt(r1);
//求生成树 memset(dis,INF,sizeof dis);
dis[r1] = ;
ddfs(r1);
//求树的直径
int maxi=r1;
for(int i=;i<=n;i++){
if(dis[i]!=INF && dis[i]>dis[maxi])maxi=i;
} memset(dis,INF,sizeof dis);
memset(vis,,sizeof vis);
dis[maxi] = ;
ddfs(maxi); int maxx=;
for(int i=;i<=n;i++){
// printf("%d ",dis[i]);
if(dis[i]!=INF)maxx=max(maxx,dis[i]);
}
printf("%d\n",(maxx+)/);
return ;
}

牛客多校第四场 A meeting 树的半径的更多相关文章

  1. 2019牛客多校第四场A meeting——树的直径

    题意: 一颗 $n$ 个节点的树上标有 $k$ 个点,找一点使得到 $k$ 个关键结点的最大距离最小. 分析: 问题等价于求树的直径,最小距离即为直径除2向上取整. 有两种求法,一是动态规划,对于每个 ...

  2. 2019牛客多校第四场 A meeting

    链接:https://ac.nowcoder.com/acm/contest/884/A来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 524288K,其他语言10485 ...

  3. 2019牛客多校第四场A meeting 思维

    meeting 题意 一个树上有若干点上有人,找出一个集合点,使得所有人都到达这个点的时间最短(无碰撞) 思路 就是找树的直径,找直径的时候记得要找有人的点 #include<bits/stdc ...

  4. 牛客多校第四场sequence C (线段树+单调栈)

    牛客多校第四场sequence C (线段树+单调栈) 传送门:https://ac.nowcoder.com/acm/contest/884/C 题意: 求一个$\max {1 \leq l \le ...

  5. 牛客多校第3场 J 思维+树状数组+二分

    牛客多校第3场 J 思维+树状数组+二分 传送门:https://ac.nowcoder.com/acm/contest/883/J 题意: 给你q个询问,和一个队列容量f 询问有两种操作: 0.访问 ...

  6. 牛客多校第四场 F Beautiful Garden

    链接:https://www.nowcoder.com/acm/contest/142/F来源:牛客网 题目描述 There's a beautiful garden whose size is n ...

  7. 牛客多校第四场 G Maximum Mode

    链接:https://www.nowcoder.com/acm/contest/142/G来源:牛客网 The mode of an integer sequence is the value tha ...

  8. 2019牛客多校第四场 I题 后缀自动机_后缀数组_求两个串de公共子串的种类数

    目录 求若干个串的公共子串个数相关变形题 对一个串建后缀自动机,另一个串在上面跑同时计数 广义后缀自动机 后缀数组 其他:POJ 3415 求两个串长度至少为k的公共子串数量 @(牛客多校第四场 I题 ...

  9. 2019年牛客多校第四场 B题xor(线段树+线性基交)

    题目链接 传送门 题意 给你\(n\)个基底,求\([l,r]\)内的每个基底是否都能异或出\(x\). 思路 线性基交板子题,但是一直没看懂咋求,先偷一份咖啡鸡板子写篇博客吧~ 线性基交学习博客:传 ...

随机推荐

  1. 利用ffmpeg进行视频软解播放

    前段时间,公司的一个项目需要一个rtsp的播放库,原本打算直接用vlc播放的,但我觉得vlc太庞大了,很多功能没必要,还不如用ffmpeg+d3d简单的实现一个库,因此就有了今天讲的这个东西.一个解码 ...

  2. UVA11427 Expect the Expected 概率dp+全概率公式

    题目传送门 题意:小明每晚都玩游戏,每一盘赢的概率都是p,如果第一盘就赢了,那么就去睡觉,第二天继续玩:否则继续玩,玩到赢的比例大于p才去睡:如果一直玩了n盘还没完成,就再也不玩了:问他玩游戏天数的期 ...

  3. Java位运算总结

    关于位运算,以前也见过,搜过,当时会用了,过后就忘了,今天好好学习一遍,然后整理一下. Java中的位运算,涉及到计算机的二进制,位用bit表示,1Byte=8bit,根据各种基本数据类型占用的字节空 ...

  4. 暑期集训日志(Day6~Day17)

    章·十七:2019-07-28:为谁辛苦为谁甜 ·昨日小结 颓爆了QAQ,昨天又垫底了. 最简单一道题弃疗的我直接被甩倒了总榜垫底…… 我……不想说啥…… 我是渣比. 我不能颓废了. 醒来啊麦克白! ...

  5. 牛客多校第三次B——线段树维护线性基交

    写线性基交函数时调试了半天.. #include<bits/stdc++.h> using namespace std; #define ll long long #define maxn ...

  6. BZOJ 3669: [Noi2014]魔法森林(lct+最小生成树)

    传送门 解题思路 \(lct\)维护最小生成树.我们首先按照\(a\)排序,然后每次加入一条边,在图中维护一棵最小生成树.用并查集判断一下\(1\)与\(n\)是否联通,如果联通的话就尝试更新答案. ...

  7. hdu多校第九场 1002 (hdu6681) Rikka with Cake 树状数组维护区间和/离散化

    题意: 在一块长方形蛋糕上切若干刀,每一刀都是从长方形某条边开始,垂直于这条边,但不切到对边,求把长方形切成了多少块. 题解: 块数=交点数+1 因为对于每个交点,唯一且不重复地对应着一块蛋糕. 就是 ...

  8. class13and14and15_登录窗口

    最终的运行效果图(程序见序号6.2): #!/usr/bin/env python# -*- coding:utf-8 -*-# ----------------------------------- ...

  9. 常用的一些 linux 指令

    1. mv linux下重命名文件或文件夹使用mv既可实现. 1.1 重命名 a.将一个名为abc.txt的文件重命名为1234.txt #mv abc.txt .txt b. 将目录A重命名为B ( ...

  10. LightOJ-1253-Misere Nim-nim博弈

    Alice and Bob are playing game of Misère Nim. Misère Nim is a game playing on k piles of stones, eac ...