【学术篇】oj.jzxx.net2701 无根树
这是一道来自OIerBBS的题目..
原帖地址:http://www.oierbbs.com/forum.php?mod=viewthread&tid=512?fromuid=71
(似乎是个很小众的BBS..不从洛谷空降根本到不了..)
(原帖的排版我忍不了。。 把题目的样式重新处理了一遍。。)
题目描述
味味最近对树很感兴趣,什么是树呢?树就是有n个节点和n-1条边形成的无环连通无向图。
味味在研究过程中想知道,对于一个无根树,当节点i作为根的时候树的高是多少。所谓树高指的是从根节点出发,到离根节点最远叶子节点所经过的节点的总数,详见输入输出样例1。味味现在遇到了一些烦心的事情,不想再继续思考了,请你帮助她解决这个问题。
输入格式
输入共N行。
第1行为一个正整数N,表示树的节点个数。
第2行到第N行里,每行两个用空格隔开的正整数a和b,表示a与b有连边。
输出格式
输出共N行,第i行表示以节点i为根时的树高。
样例输入
样例1:
3
1 2
2 3
样例2:
4
1 4
2 4
3 4
样例输出
样例1:
3
2
3
样例2:
3
3
3
2
数据范围(听LZ说的):对于100%的数据,1<=n<=1000
然而我认为的数据范围:应该是 1<=n<=5000000 (强行不和善)
(然而这样排版还是很丑)(逃
来源:oj.jzxx.net(原LZ并没有给哪个题.._ (:з」∠) _ 于是去了那个OJ.. 应该是2701..)
题目の传送门:http://oj.jzxx.net/problem.php?id=2701
(这个OJ我都不会用。。。。。。我真傻,真的)
讨论区中的做法:
法1. 暴力枚举每个点为根,搜索树的最大深度 (复杂度:O(n^2))
法2. 平衡树?(其实不知道能不能,其实我没看懂,但应该可以) (复杂度:均摊O(nlogn))
法3. 分块(这个看上去挺靠谱的?编程复杂度低于上面) (复杂度:O(n^1.5))
==========================我是学术的分割线==========================
题目大意
给定一棵无根树,求以每个点为根形成的树的树高。。
题目分析
哦?这不是等价于求树上每个点到树上其他点距离的最大值嘛。。
这就用到了一个知识点:树上任一点到树直径(*)的其中一个端点的距离最远~~
*:直径:树的直径是指树的最长简单路。
所以……这题做完了。(逃
先求直径,然后分别从直径的两端点bfs,对于求得的两个距离取最大值即可。。
直径怎么求?
两遍bfs。先随便选一个点做起点bfs找到距它距离最大的点,再从那个点进行bfs,则这次bfs找到的最长路即为树的直径(即这次找到的距它距离最大的点就是直径的另一个端点)
原理: 设起点为u,第一次bfs找到的终点v一定是树的直径的一个端点(根据某知识点),然后再bfs找到的显然是另一个端点。。
然后分别以直径的两个端点为起点进行bfs遍历树,更新每个节点的距离,然后从两距离中取最大值即可。
细心的你一定发现了,求直径时的第二遍bfs已经是从直径的一个端点出发了,我们完全可以顺带着更新路上的值了,这样我们又少了一遍bfs,这真令人高兴O(∩_∩)O
So,Obviously,经过以上步骤,我们便得到了解答。
复杂度?3遍bfs,均为O(m),因为是树,且边无向,m=(n-1)* 2,所以也就6 *(n-1)吧
总复杂度:O(n)级别(线性)
完美~~
代码实现
STL重度依赖症的代码..(不要问我为什么这三个bfs长得辣么像)
#include <queue>
#include <cstdio>
#include <cstring>
using std::vector;
using std::queue;
#define gc getchar()
#define cl(a,b) memset(a,b,sizeof(a))
const int N=1008;
vector<int> e[N];
queue<int> q;
int d[N],_d[N];
inline int gnum(){
int a=0;char c=gc;bool f=0;
for(;(c<'0'||c>'9')&&c!='-';c=gc);
if(c=='-') f=1,c=gc;
for(;c>='0'&&c<='9';c=gc) a=(a<<1)+(a<<3)+c-'0';
if(f) return ~a+1; return a;
}
int bfs1(){ //找一个端点
cl(d,-1); d[1]=0; int mi=1; q.push(1);
while(!q.empty()){
int x=q.front(); q.pop();
for(int i=0;i<e[x].size();i++){
int y=e[x][i];
if(d[y]<0){
d[y]=d[x]+1; mi=y; q.push(y);
//因为是树形结构,所以不需要像真的最短路一样加判断,一旦搜到一定是最远的(之一)
}
}
}
return mi;
}
int bfs2(int s){ //找另一个端点,更新距这个端点的距离
cl(d,-1); d[s]=1; int mi=s; q.push(s);
while(!q.empty()){
int x=q.front(); q.pop();
for(int i=0;i<e[x].size();i++){
int y=e[x][i];
if(d[y]<0){
d[y]=d[x]+1; mi=y; q.push(y);
}
}
}
return mi;
}
void bfs3(int s){ //更新距另一个端点的距离
cl(_d,-1); _d[s]=1; q.push(s);
while(!q.empty()){
int x=q.front(); q.pop();
for(int i=0;i<e[x].size();i++){
int y=e[x][i];
if(_d[y]<0){
_d[y]=_d[x]+1; q.push(y);
if(_d[y]>d[y]) d[y]=_d[y];
//偷懒ing..要是这个距离大于从另一端的距离就直接覆盖掉..
//反正树形结构一个点只会跑一次..
}
}
}
}
int main(){
int n=gnum();
for(int i=1;i<n;i++){
int u=gnum(),v=gnum();
e[u].push_back(v);
e[v].push_back(u);
}
int x=bfs1();int y=bfs2(x);bfs3(y);
for(int i=1;i<=n;i++)
printf("%d\n",d[i]);
}
//这题真的好简单呢..
写在后面
基本就是这样了,下面为好学的你们附上关于某求直径方式正确性的证明:
证明: 1) 如果u是直径上的点,则v显然是直径的终点(因为如果v不是的话,则必定存在另一个点w使得u到w的距离更长,则于BFS找到了v矛盾)
2) 如果u不是直径上的点,则u到v必然于树的直径相交(反证),那么交点到v 必然就是直径的后半段了
所以v一定是直径的一个端点,所以从v进行BFS得到的一定是直径长度
其实最重要的是理解某奇妙的知识点
The End.
【学术篇】oj.jzxx.net2701 无根树的更多相关文章
- 【学术篇】51nod 1238 最小公倍数之和
这是一道杜教筛的入(du)门(liu)题目... 题目大意 求 \[ \sum_{i=1}^n\sum_{j=1}^nlcm(i,j) \] 一看就是辣鸡反演一类的题目, 那就化式子呗.. \[ \s ...
- 【学术篇】一些水的不行的dp
最近做了几道非常水非常水的dp...... 之后刷的一些水dp也会写在这里...... 此篇题目难度不递增!!! Emmmm....... 1.luogu1043数字游戏 以前看过这个题几遍,没做这个 ...
- 【学术篇】luogu2184贪婪大陆
题目在这里哦, 戳一下就可以了~ 题目大意: 支持两种操作,区间添加一种新元素,查询区间颜色种数.. 题目标签是线段树啊,我也本来想写一个线段树,后来写不出来……(我太弱了orz) 然后就草率地看了看 ...
- 【学术篇】SPOJ-DISQUERY
一道傻逼链剖我TM总共差不多写了一小时,调了将近一天!!!!!! 题目传送门:http://www.spoj.com/problems/DISQUERY/ 嗯,偷偷递小广告:SPOJ是个挺好的OJ ( ...
- 【学术篇】bzoj2440 [中山市选2011]完全平方数
-题目の传送门- 题目大意: 找到第k个无平方因子数. 看到数据范围很大, 我们要采用比\(O(n)\)还要小的做法. 考虑如果前\(x\)个数中有\(k-1\)个无平方因子数, 而前\(x+1\)个 ...
- 【学术篇】The Xuanku Inversion Magic学习笔记
退役之前写的 然后因为退役就咕咕咕了... 后来发现数学考试能用的到个鬼就发布出来了QwQ 主要是方便自己没登录的时候查阅... 显然子集什么的是没有学会的QwQ 所以学OI的话不要看本文!!!!!& ...
- 【学术篇】CF932E Team Work && bzoj5093 图的价值
两个题的传送门 对于CF这道题, 分别考虑每种可能的集合大小, 每个大小为\(k\)的集合数量有\(\binom nk\)个, 所以最后的答案就是 \[\sum_{i=0}^n\binom{n}{i} ...
- 【学术篇】CF833B TheBakery 分治dp+主席树
题目の传送门~ 题目大意: 将\(n\)个蛋糕分成恰好\(k\)份, 求每份中包含的蛋糕的种类数之和的最大值. 这题有两种做法. 第一种是线段树优化dp, 我还没有考虑. 另一种就是分治+主席树. 然 ...
- 【学术篇】NOI2015 品酒大会 后缀数组+并查集
省选前大致是刷不了几道题了... 所以就找一些裸一点的题目练练板子算了= = 然而这题一点都不裸, 也并不怎么好写... 于是就浪费了将近一下午的时间... 然而还不是因为后缀数组板子不熟= = 首先 ...
随机推荐
- (转)python资料汇总(建议收藏)零基础必看
摘要:没料到在悟空问答的回答大受欢迎,为方便朋友,重新整理汇总,内容包括长期必备.入门教程.练手项目.学习视频. 一.长期必备. 1. StackOverflow,是疑难解答.bug排除必备网站,任何 ...
- sql准确判断某个ip
问题:如图 当我执行sql要准确查找某个IP是属于哪个库室时候,我刚开始是这样写的 select * from Definition_Read_Room where HFIP like '%172.2 ...
- 2. Vim 概念扫盲
Frm: http://www.linuxidc.com/Linux/2013-05/84031p2.htm 了解Vim的三个基本模式 当我们安装完一个编辑器后,肯定会打开它,然后在里面输入点什么东西 ...
- 基于SpringBoot的Swagger2快速入门
1. Springboot 集成 Swagger2 1.1 导入Swagger2 依赖 <!-- https://mvnrepository.com/artifact/io.springfox/ ...
- IK的整个分词处理过程
首先,介绍一下IK的整个分词处理过程: 1. Lucene的分词基类是Analyzer,所以IK提供了Analyzer的一个实现类IKAnalyzer.首先,我们要实例化一个IKAnalyzer,它有 ...
- NEO4J亿级数据全文索引构建优化
NEO4J亿级数据全文索引构建优化 一.数据量规模(亿级) 二.构建索引的方式 三.构建索引发生的异常 四.全文索引代码优化 1.Java.lang.OutOfMemoryError 2.访问数据库时 ...
- CTR预估的常用方法
1.CTR CTR预估是对每次广告的点击情况做出预测,预测用户是点击还是不点击. CTR预估和很多因素相关,比如历史点击率.广告位置.时间.用户等. CTR预估模型就是综合考虑各种因素.特征,在大量历 ...
- 廖雪峰Java16函数式编程-2Stream-5filter
1.filter简介 Stream.filter()是一个转换方法,把一个Stream转换为另一个Stream. 所谓filter操作,就是对一个Stream的所有元素进行测试,不满足条件的元素就被过 ...
- 廖雪峰Java15JDBC编程-3JDBC接口-5JDBC连接池
1. JDBC连接池 1.1 JDBC连接池简介 线程池可以复用一个线程,这样大量的小任务通过线程池的线程执行,就可以避免反复创建线程带来的开销. 同样JDBC可以复用一个JDBC连接 JDBC的连接 ...
- SQL Server Download
{ https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads }