每日一题之 hiho1787 道路建设 (树的直径)

描述
H 国有 n 座城市和 n-1 条无向道路,保证每两座城市都可以通过道路互相到达。现在 H 国要开始施工,施工分若干个阶段,第 i 个阶段会建设无向道路 (x,y) ,当且仅当存在一个数 z,满足 x ≠ z, x ≠ y, z ≠ y,且在第 i-1 个阶段后,存在无向道路 (x,z), (z,y).

现在 H 国的国王想知道,在几个阶段后,每两个不同的城市之间都有一条无向道路.

输入
第一行一个正整数 n

接下来 n-1 行,每行两个正整数 (x,y),描述一开始的一条无向道路 (x,y)

1 ≤ n ≤ 105

输出
输出最少几个阶段后,每两个不同的城市之间都有一条无向道路.

样例输入
3
1 2
2 3
样例输出
1

思路:

可以发现:一开始只有树上距离小于等于 1 的点对 (x,y) 之间有道路

然后第 1 个阶段后,所有树上距离小于等于 2 的点对 (x,y) 之间也有道路了

同理,第 i 个阶段后,所有树上距离小于等于 2i 2 i 的点对 (x,y) 之间都有道路了

于是我们只需要求出树的直径的长度 d d ,暴力求出几个阶段后有2i>=d, 主要是怎么求树的直径。

假设以u为起点,用dfs或者bfs遍历这棵树,找到距离u最远的点s,然后再以s为起点遍历这棵树,找到最远的点t,点 s,t的距离就是树的直径

证明:
假设 s-t这条路径为树的直径,或者称为树上的最长路

现有结论,从任意一点u出发搜到的最远的点一定是s、t中的一点,然后在从这个最远点开始搜,就可以搜到另一个最长路的端点,即用两遍广搜就可以找出树的最长路

证明:

1 设u为s-t路径上的一点,结论显然成立,否则设搜到的最远点为T则

dis(u,T) >dis(u,s) 且 dis(u,T)>dis(u,t) 则最长路不是s-t了,与假设矛盾

2 设u不为s-t路径上的点

首先明确,假如u走到了s-t路径上的一点,那么接下来的路径肯定都在s-t上了,而且终点为s或t,在1中已经证明过了

所以现在又有两种情况了:

1:u走到了s-t路径上的某点,假设为X,最后肯定走到某个端点,假设是t ,则路径总长度为dis(u,X)+dis(X,t)

2:u走到最远点的路径u-T与s-t无交点,则dis(u-T) >dis(u,X)+dis(X,t);显然,如果这个式子成立,

则dis(u,T)+dis(s,X)+dis(u,X)>dis(s,X)+dis(X,t)=dis(s,t)最长路不是s-t矛盾

#include <cstdio>
#include <iostream>
#include <cstring>
#include <vector>

using namespace std;

const int maxn = 1e5 + 5;

vector<int>g[maxn];
int d[maxn];
bool vis[maxn];

void dfs(int u)
{
    vis[u] = 1;
    for (auto &v: g[u]) {
        if (vis[v]) continue;
        d[v] = d[u]+1;
        dfs(v);
    }
}

int main()
{
    memset(d,0,sizeof(d));
    memset(vis,0,sizeof(vis));
    int n,u,v;
    cin >> n;
    for (int i = 1; i < n; ++i) {
        cin >> u >> v;
        g[u].push_back(v);
        g[v].push_back(u);
    }

    dfs(1);
    int len = 0,st;

    for (int i = 1; i <= n; ++i) {
        if (d[i] > len) {
            st = i;
            len = d[i];
        }
    }
    memset(d,0,sizeof(d));
    memset(vis,0,sizeof(vis));
    len = 0;
    dfs(st);
    for (int i = 1; i <= n; ++i)
        len = max(len,d[i]);

    int k = 1;
    int res = 0;
    while(k < len) {
        k *= 2;
        ++res;
    }
    cout << res << endl;

    return 0;
}

Python网络爬虫与推荐算法新闻推荐平台:网络爬虫:通过Python实现新浪新闻的爬取,可爬取新闻页面上的标题、文本、图片、视频链接(保留排版) 推荐算法:权重衰减+标签推荐+区域推荐+热点推荐.zip项目工程资源经过严格测试可直接运行成功且功能正常的情况才上传,可轻松复刻,拿到资料包后可轻松复现出一样的项目,本人系统开发经验充足(全领域),有任何使用问题欢迎随时与我联系,我会及时为您解惑,提供帮助。 【资源内容】:包含完整源码+工程文件+说明(如有)等。答辩评审平均分达到96分,放心下载使用!可轻松复现,设计报告也可借鉴此项目,该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的。 【提供帮助】:有任何使用问题欢迎随时与我联系,我会及时解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 下载后请首先打开README文件(如有),项目工程可直接复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值