题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5374

思路:题目的意思是求树上的两点,使得树上其余的点到其中一个点的最长距离最小。可以想到这题与树直径有关,我们可以这样做,首先求出树的直径,然后取出树的中点以及与该中点相邻,并且是直径上的一个点,这样就把这棵树划分为两颗子树,然后分别求出这两棵树的直径,最后要选择的两个点分别就是这两棵树的直径上的中点。

一开始是用dfs写的,结果爆栈了,改成bfs就过了。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std; const int MAX_N = (200000 + 20000);
struct Edge {
int v, w, next;
} edge[MAX_N << 1]; int N, NE, head[MAX_N];
void Init()
{
NE = 0;
memset(head, -1, sizeof(head));
} void Insert(int u, int v, int w)
{
edge[NE].v = v;
edge[NE].w = w;
edge[NE].next = head[u];
head[u] = NE++;
} int dep[MAX_N], path[MAX_N], st, ed, s_mid, e_mid;
int ans_minDist, ans_point1, ans_point2;
bool vis[MAX_N]; bool check(int u, int v)
{
if (u == s_mid && v == e_mid) return true;
if (u == e_mid && v == s_mid) return true;
return false;
} void bfs(int u, int fa, int deep)
{
dep[u] = deep;
path[u] = fa;
vis[u] = true;
queue<int > que;
que.push(u); while (!que.empty()) {
int u = que.front();
que.pop(); for (int i = head[u]; ~i; i = edge[i].next) {
int v = edge[i].v, w = edge[i].w;
if (v == fa || check(u, v) || vis[v]) continue;
dep[v] = dep[u] + w;
path[v] = u;
vis[v] = true;
que.push(v);
}
}
} void gao()
{
s_mid = e_mid = -1;
memset(vis, false, sizeof(vis));
bfs(1, -1, 0); int max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (dep[i] > max_deep) max_deep = dep[i], st = i;
} memset(vis, false, sizeof(vis));
bfs(st, -1, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (dep[i] > max_deep) max_deep = dep[i], ed = i;
} int tmp = ed, cnt = 0;
while (tmp != -1) {
tmp = path[tmp];
++cnt;
if (cnt == max_deep / 2) s_mid = tmp;
else if (cnt == max_deep / 2 + 1) e_mid = tmp;
}
} void solve()
{
//get point1
memset(vis, false, sizeof(vis));
bfs(s_mid, e_mid, 0); int max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i], st = i;
} memset(vis, false, sizeof(vis));
bfs(st, -1, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i], ed = i;
} int tmp = ed, cnt = 0;
ans_point1 = ed;
while (tmp != -1) {
tmp = path[tmp];
++cnt;
if (cnt == max_deep / 2) ans_point1 = tmp;
} memset(vis, false, sizeof(vis));
bfs(ans_point1, -1, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i];
} ans_minDist = max_deep; //get point2
memset(vis, false, sizeof(vis));
bfs(e_mid, s_mid, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i], st = i;
} memset(vis, false, sizeof(vis));
bfs(st, -1, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i], ed = i;
} tmp = ed, ans_point2 = ed, cnt = 0;
while (tmp != -1) {
tmp = path[tmp];
++cnt;
if (cnt == max_deep / 2) ans_point2 = tmp;
} memset(vis, false, sizeof(vis));
bfs(ans_point2, -1, 0); max_deep = -1;
for (int i = 1; i <= N; ++i) {
if (vis[i] && dep[i] > max_deep) max_deep = dep[i];
} ans_minDist = max(ans_minDist, max_deep); } int main()
{
int Cas;
scanf("%d", &Cas);
while (Cas--) {
scanf("%d", &N); Init();
for (int i = 1; i < N; ++i) {
int u, v;
scanf("%d %d", &u, &v);
Insert(u, v, 1);
Insert(v, u, 1);
} if (N == 2) {
puts("0 1 2");
continue;
} gao();
solve(); printf("%d %d %d\n", ans_minDist, ans_point1, ans_point2);
}
}


zoj 3820(2014牡丹江现场赛B题)的更多相关文章

  1. zoj 3827(2014牡丹江现场赛 I题 )

    套公式 Sample Input 33 bit25 25 50 //百分数7 nat1 2 4 8 16 32 3710 dit10 10 10 10 10 10 10 10 10 10Sample ...

  2. zoj 3819(2014牡丹江现场赛 A题 )

    题意:给出A班和B班的学生成绩,如果bob(A班的)在B班的话,两个班级的平均分都会涨.求bob成绩可能的最大,最小值. A班成绩平均值(不含BOB)>A班成绩平均值(含BOB) &&a ...

  3. 2014 牡丹江现场赛 i题 (zoj 3827 Information Entropy)

    I - Information Entropy Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%lld & %l ...

  4. ZOJ 3822 ( 2014牡丹江区域赛D题) (概率dp)

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5376 题意:每天往n*m的棋盘上放一颗棋子,求多少天能将棋盘的每行每列都至少有 ...

  5. 2014 牡丹江现场赛 A.Average Score(zoj 3819) 解题报告

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5373 题目意思: 有两个class:A 和 B,Bob 在 Clas ...

  6. ZOJ 2819 Average Score 牡丹江现场赛A题 水题/签到题

    ZOJ 2819 Average Score Time Limit: 2 Sec  Memory Limit: 60 MB 题目连接 http://acm.zju.edu.cn/onlinejudge ...

  7. 2014牡丹江 现场赛 F zoj 3824 Fiber-optic Network

    首先赞一下题目, 好题 题意: Marjar University has decided to upgrade the infrastructure of school intranet by us ...

  8. zoj 3809 枚举水题 (2014牡丹江网赛 A题)

    题目大意:给出一列取样的几个山的高度点,求山峰有几个? Sample Input 291 3 2 4 6 3 2 3 151 2 3 4 5Sample Output 30 # include < ...

  9. ACM学习历程——ZOJ 3822 Domination (2014牡丹江区域赛 D题)(概率,数学递推)

    Description Edward is the headmaster of Marjar University. He is enthusiastic about chess and often ...

随机推荐

  1. 国内优秀npm镜像推荐及使用

    npm全称Node Package Manager,是node.js的模块依赖管理工具.由于npm的源在国外,所以国内用户使用起来各种不方便.下面整理出了一部分国内优秀的npm镜像资源,国内用户可以选 ...

  2. TypeScript Declaration Merging(声明合并)

    TypeScript中有一些独特的概念,来自需要描述JavaScript对象类型发生了哪些变化.举个例子,最为独特的概念就是"声明合并".理解了这个概念将会对你在当前JavaScr ...

  3. 如何同时打开两个excel

    1. 打开一个excel1 2. 不要双击想要打开的excel2.右键excel应用的图标,选择excel2007. 3. 将excel2拖动到2所打开的新建excel中. 4. over.

  4. QQ等级表

    什么是QQ等级呢? 2003年,腾讯公司推出了QQ等级制度 . 最早是以小时,来计算的,那段时间,绝大部分QQ用户都在挂QQ,之后就有不少媒体指责其浪费能源,在有关部门的介入下,腾讯公司将QQ等级变为 ...

  5. Python之路【第二十一篇】Django ORM详解

    ORM回顾 关系对象映射(Object Relational Mapping,简称ORM). django中遵循 Code Frist 的原则,即:根据代码中定义的类来自动生成数据库表. 对于ORM框 ...

  6. 判断来防ip是否为蜘蛛

    判断网站来防IP是否为蜘蛛,用命令查询 :     一.在windows平台 蜘蛛反查命令:nslookup IP 点击"开始"-"运行"-"cmd& ...

  7. 如何自定义iphone个性铃音

    准备工作:itunes.(Netease Cloud Music).iphone 1.下载你想要的铃音原音乐: 2.打开itunes,向音乐库中添加刚刚下载的音乐: "文件"-&g ...

  8. R语言-用R眼看琅琊榜小说的正确姿势

    博客总目录:http://www.cnblogs.com/weibaar/p/4507801.html 目录: 零:写在前面的一些废话 一.R眼看琅琊榜的基本原理 1.导入数据 2.筛选数据 3.多条 ...

  9. XML Schema and XMLspy notes

    Introduction An xml documents consists of elements, attributes and text. There are two structures in ...

  10. android 项目中如何引入第三方jar包

    http://www.360doc.com/content/13/0828/08/11482448_310390794.shtml