这好像很裸的样子,题目说了一堆废话,最后其实就是让求三个点,使得\(AB+BC\)最小,感性的理解蒙一下,应该有一条边是直径,不然把任意一条边换成直径都可以使答案更优,然后就找直径呗,找完直径枚举直径两端点到其他点的距离,最后答案为直径长+\(max(min(dis1,dis2))\),好像就水过这道题了?
由于它是个树,所以根据某学长的理论,随便跑都可以,我dfs忘记咋写的了,想练一下spfa来,毕竟我那个不熟,但看了看时间太晚了就打了一个dij,没想到真的过了。。。证明的话明天心情好再补吧
#include<iostream>
#include<queue>
#define ll long long
using namespace std;
const int N=2e5+10;
struct Edge{
int to,nxt;
ll val;
}e[N<<1];
struct Node{
int id;ll val;
Node(){}
Node(int a,ll b){id=a,val=b;}
bool operator <(const Node&A)const {
return val>A.val;
}
};
int Head[N],len;
void Ins(int a,int b,ll c){
e[++len].to=b;e[len].val=c;
e[len].nxt=Head[a];Head[a]=len;
}
int n;bool inq[N];
ll dis1[N],dis2[N];
int dij(int s,ll dis[N]){
for(int i=1;i<=n;i++){
dis[i]=1e17;
inq[i]=0;
}
dis[s]=0;inq[s]=1;
priority_queue<Node> q;q.push(Node(s,0));
while(!q.empty()){
Node u=q.top();q.pop();inq[u.id]=0;
for(int i=Head[u.id];i;i=e[i].nxt){
int v=e[i].to;
if(inq[v])continue;
inq[v]=1;
if(dis[v]>dis[u.id]+e[i].val){
dis[v]=dis[u.id]+e[i].val;
q.push(Node(v,dis[v]));
}
}
}
ll Max=0;int id=0;
for(int i=1;i<=n;i++)
if(dis[i]>Max)Max=dis[i],id=i;
return id;
}
int main(){
ios::sync_with_stdio(false);
int m;
cin>>n>>m;
for(int i=1;i<=m;i++){
int a,b;ll c;
cin>>a>>b>>c;
Ins(a,b,c);Ins(b,a,c);
}
int a=dij(1,dis1);
int b=dij(a,dis2);
dij(b,dis1);
ll Max=0;
for(int i=1;i<=n;i++)
Max=max(Max,min(dis1[i],dis2[i]));
cout<<dis2[b]+Max;
}
tips:longlong数组用memset会出问题不知道为啥
原文:https://www.cnblogs.com/anyixing-fly/p/12663882.html