题意:

n个点,m条边,m <= n <= 100000,边的长度都为1。

点从 0 ~ n-1 编号。开始时图是不连通的,并且没有环。

通过加入一些边后,可以使图连通。要求加入的边不能多余(即生成的图是一棵树)。

问连通后的图,任意两点之间的距离的最大值,最小可以是多少?

既然刚开始图不连通也无环,那么就是一些树(特殊情况是点)。

于是题目就变成了,如何把很多棵树连起来,使最后生成的树直径最小。

可以想到,如果把两棵直径为 a 和 b 的树加一条边连成一棵,那么直径最小的新树直径为 (a+1)/2 + (b+1)/2 + 1 (两棵树的直径 / 2,向上取整,的和再加 1)

选择其中一个直径最大的子树,遍历所有其他的子树,然后将其他的子树加到这个子树上面。

求树的直径可以用DP。

这道题场上很快想出来了做法。然后一直T在 test 22。

原因是不会写树的直径DP求法,以及,memset超时。

每次找到新的联通块(新的一棵树)求树的直径就要memset一遍是不需要的。因为那些点肯定是不冲突的。

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <algorithm>
  4. #include <iostream>
  5. #include <queue>
  6. #include <cmath>
  7. #include <stack>
  8. #include <set>
  9. #include <map>
  10.  
  11. using namespace std;
  12. const int maxn = + ;
  13.  
  14. int fa[maxn], dis[maxn], tot = , v[maxn];
  15. bool vis[maxn];
  16.  
  17. struct Node
  18. {
  19. int v, next, last, z;
  20. }a[maxn];
  21.  
  22. void build(int x, int y)
  23. {
  24. tot++;
  25. a[tot].v = y;
  26. a[tot].next = a[x].last;
  27. a[x].last = tot;
  28. }
  29.  
  30. void dp(int x, int &ans)
  31. {
  32. v[x] = ;
  33. for (int tmp = a[x].last; tmp; tmp = a[tmp].next)
  34. {
  35. int y = a[tmp].v;
  36. if (v[y]) continue;
  37. dp(y, ans);
  38. ans = max(ans, dis[x] + dis[y] + );
  39. dis[x] = max(dis[x], dis[y] + );
  40. }
  41. }
  42.  
  43. void DFS(int x)
  44. {
  45. vis[x] = true;
  46. for (int tmp = a[x].last; tmp; tmp = a[tmp].next)
  47. {
  48. int y = a[tmp].v;
  49. if (!vis[y]) DFS(y);
  50. }
  51. }
  52.  
  53. int main()
  54. {
  55. int n, m;
  56. scanf("%d%d", &n, &m);
  57.  
  58. int x, y;
  59. for (int i = ; i <= m; i++)
  60. {
  61. scanf("%d%d", &x, &y);
  62. build(x, y);
  63. build(y, x);
  64. }
  65.  
  66. memset(vis, false, sizeof(vis));
  67.  
  68. memset(v, , sizeof(v));
  69. memset(dis, , sizeof(dis));
  70. int q[maxn], tt = ;
  71. for (int i = ; i < n; i++)
  72. if (!vis[i])
  73. {
  74. int t = ;
  75. dp(i, t);
  76. q[++tt] = t;
  77. DFS(i);
  78. }
  79.  
  80. int ans = , flag = ;
  81. for (int i = ; i <= tt; i++)
  82. if (q[i] > ans)
  83. {
  84. ans = q[i];
  85. flag = i;
  86. }
  87.  
  88. for (int i = ; i <= tt; i++)
  89. if (i != flag)
  90. ans = max(ans, (q[i]+)/+(ans+)/ + );
  91. printf("%d\n", ans);
  92.  
  93. return ;
  94. }

Gym - 100781A Adjoin the Networks (树的直径)的更多相关文章

  1. 【DFS】Gym - 100781A - Adjoin the Networks

    给你一个森林,让你把它连接成一颗树,使得直径最小. 就求出每颗树的重心以后,全都往直径最大的那个的重心上连,一般情况是最大/2+次大/2+1,次大/2+第三大/2+2 中取较大者. 还有些特殊情况要特 ...

  2. codeforces GYM 100781A【树的直径】

    先求出每棵树的直径,排个序,要想图的直径最小的话需要每棵树的直径中点像直径最大的树的直径中点连边,这样直径有三种情况:是直径最大的树的直径:a[tot]:是直径最大的树和直径第二大的树的半径拼起来+1 ...

  3. codeforces GYM 100114 J. Computer Network 无相图缩点+树的直径

    题目链接: http://codeforces.com/gym/100114 Description The computer network of “Plunder & Flee Inc.” ...

  4. codeforces GYM 100114 J. Computer Network tarjan 树的直径 缩点

    J. Computer Network Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100114 Des ...

  5. Gym - 100676H Capital City(边强连通分量 + 树的直径)

    H. Capital City[ Color: Black ]Bahosain has become the president of Byteland, he is doing his best t ...

  6. Gym - 100676H H. Capital City (边双连通分量缩点+树的直径)

    https://vjudge.net/problem/Gym-100676H 题意: 给出一个n个城市,城市之间有距离为w的边,现在要选一个中心城市,使得该城市到其余城市的最大距离最短.如果有一些城市 ...

  7. CF GYM 100781A(菊花图+直径)

    题目大意 给出若干颗树用最少的边把它们连成一个无向连通图,同时使图的直径最小.输出最小直径. 题解 我们定义树的半径为(树的直径+1)/2.符合题意的连接方式为.所有树的“中点”连在直径最长的树的中点 ...

  8. hiho 1050 树的直径

    #1050 : 树中的最长路 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 上回说到,小Ho得到了一棵二叉树玩具,这个玩具是由小球和木棍连接起来的,而在拆拼它的过程中, ...

  9. 牡丹江.2014B(图论,树的直径)

    B - Building Fire Stations Time Limit:5000MS     Memory Limit:131072KB     64bit IO Format:%lld & ...

随机推荐

  1. Hadoop实战项目:小文件合并

    项目背景 在实际项目中,输入数据往往是由许多小文件组成,这里的小文件是指小于HDFS系统Block大小的文件(默认128M),早期的版本所定义的小文件是64M,这里的hadoop-2.2.0所定义的小 ...

  2. 常见的JedisConnectionException 异常

    最近在使用redis出现以下的异常: 1.redis.clients.jedis.exceptions.JedisConnectionException: java.net.ConnectExcept ...

  3. readonly与const的区别

    readonly 关键字与 const 关键字不同.const 字段只能在该字段的声明中初始化.readonly字段可以在声明或构造函数中初始化.因此,根据所使用的构造函数,readonly字段可能具 ...

  4. mongodb关联查询 和spring data mongodb

    GITHUB:https://github.com/peterowang/Springdata-mongo 使用DBRefs DBRefs中有三个字段 - $ref - 此字段指定引用文档的集合 $i ...

  5. vue-quill-editor上传内容由于图片是base64的导致字符太长的问题解决

    vue-quill-editor是个较为轻量级富文本框,相较于ueditor,开发更编辑,更加直观,如果大家伙在需求允许的情况下,还是会比较建议使用vue-quill-editor vue-quill ...

  6. Servlet中文件上传的几种方式

    上传文件,因为上传的都是二进制数据,所以在Servlet中就不能直接用request.getParameter();方法进行数据的获取,需要借助第三方jar包对上传的二进制文件进行解析.常用的方式如下 ...

  7. 移动端的300ms延迟和点击穿透

    移动端300ms延迟:假定这么一个场景.用户在 浏览器里边点击了一个链接.由于用户可以进行双击缩放或者双击滚动的操作,当用户一次点击屏幕之后,浏览器并不能立刻判断用户是确实要打开这个链接,还是想要进行 ...

  8. js字符串、数组、时间、日期对象

    js对字符串.数组.日期的操作是在以后项目中频繁使用的操作,所以呢....所以大家看着办,其实并不难哈,就是有点无聊,我承认这是我百度的,哈哈哈哈 <!DOCTYPE html><h ...

  9. jsp之初识UserBean

    package com.java.model; public class Student { private String name; private int age; public String g ...

  10. db2新添用户

    --1.新添用户  -目录 /XX/XX  -组  XX 用户名useradd -d /home/xx -g users xx--2.修改密码passwd xx--3.在QC中grant权限.新添表空 ...