题意

一个城市由节点和连接节点的街道组成,街道是双向的。

此刻大雪覆盖了这个城市,市长确定了一些街道要将它们清扫干净,这些街道保证所有的节点可以通过它们连通而且街道数目尽可能小。

现有两台相同的扫雪机\(S\)和\(M\),它们的起点在同一个节点上。

所有被确定的街道必须至少被一台扫雪机经过,才能完成清扫任务,完成任务后\(S\)和\(M\)可以在原地停下,不必集合到某一点。

扫雪机的行进是需要耗费油量的(即使扫雪机行驶的是已被扫净的街道),因此扫雪机行进的总距离越小越好,你需要计算两台扫雪机完成任务的最小总行进距离。

分析

题目的条件暗示了这是一棵树,问题来了,题意的最短距离(两个扫雪机的情况下)意味着什么?

我们先考虑一个扫雪机的情况。这样,从任意一个地方开始,然后转化为以其为根的有根树,遍历它的所有子树,完事了。最后一个“最长”的子树不用回来,这样在最优的情况下走多久呢?2*所有边长-全图的最长边——也就是图的直径,这是它的定义。

那么两个机子意味着什么呢?很意外,其实还是这个答案,哈哈。为什么?因为既然转化为以直径的端点为根(见上最优的情况)的有根树了,那么这种情况下根结点必只有一个子节点。那么无论怎么走,他们仅仅只能省去走直径的“回程”,在子树里他们是无论如何要走一遍回来的,这样,与一个扫雪机的情况是一致的。

参见:https://www.cnblogs.com/dilthey/p/7231438.html

接下来解决树的直径问题——树的最长边。

做法是这样的:

从随便一个点(记为\(w\))出发,到一个节点结束,记这个节点为\(u\);然后从这个点出发,到另一个点完事,记这个节点为\(v\),那么长度就是u到v的长度。

参见:https://blog.csdn.net/tc_to_top/article/details/47002255

这样为什么能work呢?

  1. 如果\(w\)是直径上的一点,那么u一定是直径的一个端点,否则从另一端点出发到达\(u\)反而比直径长了,不合题意。
  2. 如果\(w\)不是的话,那它到最远的点必经过一直径上的点(记为\(c\)),后同(1)的情况。为什么?如果不经过直径的一点,此时有\(dis[w,u']=dis[w,x]+dis[x,u']>dis[w,t]+dis[t,u]\),其中\(t\)是直径上的一点,\(x\)是\(w\)到\(t\)的一点。那么\(dis[x,u']>dis[x,t]+dis[t,u]\),也就有\(dis[v,t]+dis[t,x]+dis[x,u']=dis[v,u']>dis[v,u]=dis[v,t]+dis[t,u]\),这就坏事了:我们明明有\((v,u)\)是树的直径的。所以得到矛盾,原先的证明不成立。

参见:https://blog.csdn.net/qq_32400847/article/details/51469917

代码

/*
* Filename: poj1849.cpp
* Date: 2018-11-03
*/ #include <cstring>
#include <vector>
#include <iostream> #define INF 0x3f3f3f3f
#define PB push_back
#define MP make_pair
#define fi first
#define se second
#define rep(i,a,b) for(repType i=(a); i<=(b); ++i)
#define per(i,a,b) for(repType i=(a); i>=(b); --i)
#define ZERO(x) memset(x, 0, sizeof(x))
#define MS(x,y) memset(x, y, sizeof(x))
#define ALL(x) (x).begin(), (x).end() #define QUICKIO \
ios::sync_with_stdio(false); \
cin.tie(0); \
cout.tie(0);
#define DEBUG(...) fprintf(stderr, __VA_ARGS__), fflush(stderr) using namespace std;
typedef int repType;
typedef long long ll; int n,s;
const int MAXN=100005;
struct Edge
{
int u,v,w;
Edge(int _u, int _v, int _w):
u(_u), v(_v), w(_w) {}
};
vector<Edge> edges;
vector<int> G[MAXN]; void add_edge(int u, int v, int w)
{
edges.PB(Edge(u,v,w));
G[u].PB(int(edges.size())-1);
} int dist[MAXN];
void dfs(int x)
{
rep(i,0,int(G[x].size())-1)
{
if(dist[edges[G[x][i]].v]==-1)
{
dist[edges[G[x][i]].v]=dist[x]+edges[G[x][i]].w;
dfs(edges[G[x][i]].v);
}
}
} int
main()
{
QUICKIO
while(cin>>n>>s)
{
edges.clear();
rep(i,1,n) G[i].clear();
ll ans=0;
rep(i,1,n-1)
{
int a,b,c;
cin>>a>>b>>c;
add_edge(a,b,c);
add_edge(b,a,c);
ans+=c;
}
MS(dist, -1);
dist[s]=0;
dfs(s);
int maxi=s;
rep(i,1,n) if(dist[maxi]<dist[i]) maxi=i;
MS(dist, -1);
dist[maxi]=0;
dfs(maxi);
rep(i,1,n)
{
if(dist[maxi]<dist[i]) maxi=i;
}
cout<<2*ans-dist[maxi]<<endl;
}
return 0;
}

「日常训练&知识学习」树的直径(POJ-1849,Two)的更多相关文章

  1. 「日常训练&知识学习」树的分块(王室联邦,HYSBZ-1086)

    题意与分析 这题的题意就是树分块,更具体的看题目(中文题). 学习这一题是为了树的分块,为树上莫队做铺垫. 参考1:https://blog.csdn.net/LJH_KOQI/article/det ...

  2. 「日常训练&知识学习」单调栈

    这几天的知识学习比较多,因为时间不够了.加油吧,为了梦想. 这里写几条简单的单调栈作为题解记录,因为单调栈的用法很简单,可是想到并转化成用这个需要一些题目的积淀. 相关博客参见:https://blo ...

  3. 「日常训练&知识学习」莫队算法(二):树上莫队(Count on a tree II,SPOJ COT2)

    题意与分析 题意是这样的,给定一颗节点有权值的树,然后给若干个询问,每次询问让你找出一条链上有多少个不同权值. 写这题之前要参看我的三个blog:Codeforces Round #326 Div. ...

  4. 「国庆训练&知识学习」图的最大独立集与拓展(Land of Farms,HDU-5556)

    题意 一个\(N*M\)的矩阵,其中"."代表空地,"0-9"代表古代建筑,我们如果选择了一个编号的古代建筑想要建立,那么对应就要将全部该编号的建筑建立起来,如 ...

  5. 「日常训练」Caterpillar(POJ-3310)

    题意与分析 一条很有趣的题目.给一个无向图,问它是否无环,且可以在上面找到一条线,使所有的顶点要么在线上要么不在线上但在与线相连的边上. 那么首先要确定所有点联系在一起.这个可以同判环一起处理:如果建 ...

  6. 「日常训练」ZgukistringZ(Codeforces Round #307 Div. 2 B)

    题意与分析(CodeForces 551B) 这他妈哪里是日常训练,这是日常弟中弟. 题意是这样的,给出一个字符串A,再给出两个字符串B,C,求A中任意量字符交换后(不限制次数)能够得到的使B,C作为 ...

  7. 「日常训练」湫湫系列故事——设计风景线(HDU-4514)

    题意与分析 中文题目,木得题意的讲解谢谢. 然后还是分解成两个任务:a)判环,b)找最长边. 对于这样一个无向图,强行转换成负权然后bellman-ford算法求最短是难以实现的,所以感谢没有环--我 ...

  8. 树的直径 poj 2631

    树的直径:从随意一点出发,BFS找到最远的距离,然后在从该点出发BFS找到最远的距离 #include <iostream> #include <algorithm> #inc ...

  9. 「日常训练」Duff in the Army (Codeforces Round #326 Div.2 E)

    题意(CodeForces 588E) 给定一棵\(n\)个点的树,给定\(m\)个人(\(m\le n\))在哪个点上的信息,每个点可以有任意个人:然后给\(q\)个询问,每次问\(u\)到\(v\ ...

随机推荐

  1. 最简单的Servlet继承HttpServlet查询数据库登录验证

    <%-- Created by IntelliJ IDEA. User: yunqing Date: 2017-12-06 Time: 9:11 To change this template ...

  2. event对象,ie8及其以下

    event 对象是 JavaScript 中一个非常重要的对象,用来表示当前事件.event 对象的属性和方法包含了当前事件的状态.当前事件,是指正在发生的事件:状态,是与事件有关的性质,如引发事件的 ...

  3. Android客户端与服务器端通过DES加密认证

    转载地址:http://blog.csdn.net/spring21st/article/details/6730283 由于Android应用没有像web开发中的session机制,所以采用PHPS ...

  4. 【转】NHIBERNATE的各种保存方式的区别 (SAVE,PERSIST,UPDATE,SAVEORUPDTE,MERGE,FLUSH,LOCK)

    前言 今天学学习NH这个框架,在新增对象的时候,看见大神用了persist而没有用Save,心中比较疑惑,查阅资料的时候,发现这篇写的非常不错,转载供大家参考. hibernate的保存hiberna ...

  5. Entity Framework中DbContext结合TransactionScope提交事务的正确方式

    问: I would like know what is the best possible way to implement transactions with DBContext. In part ...

  6. python 面向对象之添加功能

    '''**#实现功能**案列 姓名:王飞 年龄:30 性别:男 工龄:5我承诺,我会认真教课.王飞爱玩象棋 姓名:小明 年龄:15 性别:男 学号:00023102我承诺,我会 好好学习.小明爱玩足球 ...

  7. [开源]JSON文本格式化工具(简码万能助手开源扩展程序)

    现在的网站大多都是使用json进行API式前后端数据交互, 有时抓包得到的是一串没格式化的JSON文本, 不太方便分析, 所以我自行写了个开源扩展程序, 可以方便地格式化JSON文本.   当然,你也 ...

  8. 洛谷P3812 【模板】线性基

    题目背景 这是一道模板题. 题目描述 给定n个整数(数字可能重复),求在这些数中选取任意个,使得他们的异或和最大. 输入输出格式 输入格式: 第一行一个数n,表示元素个数 接下来一行n个数 输出格式: ...

  9. [MYSQL][2]索引

    创建表时创建索引: 在已经存在的表上创建索引: 方法一 方法二 删除索引:

  10. HTML5中的拖拽与拖放(drag&&drop)

    1.drag 当拖动某个元素时,将会依次触发下列事件: 1)dragstart:按下鼠标键并开始移动鼠标时,会触发该事件 2)drag:dragstart触发后,随即便触发drag事件,而且在元素被拖 ...