【图论】最小直径生成树
【图论】最小直径生成树
题目描述
求图中的一个生成树,使得直径最小,求这个直径。
算法描述
考虑求出一个绝对中心,这个中心是一个点,与生成树上所有点距离最大值最小,这个点可能在一条边上。
考虑枚举一条边
那么对于所有的点,这个距离取最大值,就是一个锯齿状的函数。
毫无疑问,最大值中取最小值就是最低的拐点,这里我们分开讨论:
-
如果这个拐点是端点,那么直接枚举顶点,看哪个点距离它最远,由于绝对中心在端点上,那么一定有另外一个点同样是最远距离,乘 2 即可。
-
如果不是端点,那么需要找到最长的直径,观察上图。
如果我们按照
所以我们记录当前扫到的
标记的
由于
综上,枚举每条边,在枚举点判断即可,点边不同阶可以用 floyd 算法,时间复杂度
如果想求中心,找到直径后取中点即可。
想求生成树,以中心为源跑最短路树即可。
#include<bits/stdc++.h>
using namespace std;
const int N = 205;
int n,dis[N][N],m,rk[N][N];
struct Edge{
int u,v,w;
}e[N * N];
typedef long double ld;
int main()
{
memset(dis,0x3f,sizeof(dis));
cin>>n>>m;
for(int i = 1,x,y,z;i <= m;i++)
{
cin>>x>>y>>z;
dis[x][y] = dis[y][x] = z;
e[i] = Edge{x,y,z};
}
for(int i = 1;i <= n;i++) dis[i][i] = 0;
for(int k = 1;k <= n;k++)
for(int i = 1;i <= n;i++)
for(int j = 1;j <= n;j++)
dis[i][j] = min(dis[i][j],dis[i][k] + dis[k][j]);
for(int i = 1;i <= n;i++)
{
for(int j = 1;j <= n;j++) rk[i][j] = j;
sort(rk[i] + 1,rk[i] + n + 1,[&](int x,int y) {return dis[i][x] < dis[i][y];});
}
ld ans = 1e18;
for(int i = 1;i <= n;i++) ans = min(ans,dis[i][rk[i][n]] * (ld)2);
for(int i = 1;i <= m;i++)
{
int u = e[i].u,v = e[i].v,p = rk[u][n];
for(int j = n;j >= 1;j--)
{
int nw = rk[u][j];
if(dis[v][nw] > dis[v][p])
{
ans = min(ans,(ld)dis[u][nw] + dis[v][p] + e[i].w);
p = nw;
}
}
}
cout<<fixed<<setprecision(10)<<ans / 2;
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!