傳送門
ios
求出一條s到t的路徑,使得這條路徑上通過最高溫度最小的前提下,總長度最短。算法
什麼最大值最小很容易想到二分...就每次枚舉最高溫度而後在這個溫度下跑最短路看是否連通。ide
可是這道題也能夠用最小生成樹的算法來作qwqspa
想要溫度儘可能小,就以溫度爲關鍵字,跑一遍最小生成樹,就獲得了使s,t連通所須要的最小的最高溫度。code
(注意,這裏最小生成樹的結束條件是fa(s)==fa(t) )blog
而後在全部溫度不高於這個最高溫度的邊中跑最短路便可。get
話說這題有多組數據啊...string
發現it
while(scanf("%d%d",&n,&m)!=EOF)
能夠寫成io
while(~scanf("%d%d",&n,&m))
代碼以下
#include<cstdio> #include<iostream> #include<cmath> #include<cstring> #define MogeKo qwq #include<algorithm> #include<queue> using namespace std; const int maxn = 2e6+10; const int INF = 0x3f3f3f3f; int n,m,s,t,cnt,num; int fa[maxn],pre[maxn],a[maxn]; int head[maxn],to[maxn],nxt[maxn]; double hot,dis[maxn],val[maxn]; bool vis[maxn]; struct edge { int u,v; double tem,len; bool operator < (const edge & N)const { return tem < N.tem; } } e[maxn]; void add(int x,int y,double z) { to[++cnt] = y; nxt[cnt] = head[x]; head[x] = cnt; val[cnt] = z; } int getfa(int x) { if(fa[x] == x)return x; return fa[x] = getfa(fa[x]); } void init() { cnt = num = 0; memset(head,0,sizeof(head)); memset(nxt,0,sizeof(nxt)); memset(pre,0,sizeof(pre)); memset(vis,0,sizeof(vis)); for(int i = 1; i <= n; i++) { fa[i] = i; dis[i] = INF; } sort(e+1,e+m+1); } void kruskal() { for(int i = 1; i <= m; i++) { int x = getfa(e[i].u); int y = getfa(e[i].v); if(x == y)continue; fa[x] = y; if(getfa(s) == getfa(t)) { hot = e[i].tem; break; } } } void addedge() { for(int i = 1; i <= m; i++) { if(e[i].tem > hot) break; add(e[i].u,e[i].v,e[i].len); add(e[i].v,e[i].u,e[i].len); } } void dijkstra(int s) { priority_queue < pair<double,int>, vector< pair<double,int> >,greater< pair<double,int> > > q; dis[s] = 0; q.push(make_pair(0,s)); while(!q.empty()) { int u = q.top().second; q.pop(); if(vis[u]) continue; vis[u] = true; for(int i = head[u]; i; i = nxt[i]) { int v = to[i]; if(dis[u]+val[i] < dis[v]) { dis[v] = dis[u]+val[i]; pre[v] = u; q.push(make_pair(dis[v],v)); } } } for(int i = t; i ; i = pre[i]) a[++num] = i; } int main() { while(~scanf("%d%d",&n,&m)) { scanf("%d%d",&s,&t); for(int i = 1; i <= m; i++) scanf("%d%d%lf%lf",&e[i].u,&e[i].v,&e[i].tem,&e[i].len); init(); kruskal(); addedge(); dijkstra(s); for(int i = num; i > 1; i--) printf("%d ",a[i]); printf("%d\n",a[1]); printf("%.1lf %.1lf\n",dis[t],hot); } return 0; }