WOJ#3836 Sightseeing Trip
描述
给定一张无向图,求图中一个至少包含 3 个点的环,环上的节点不重复,并且环上的边的长度之和最小。该问题称为无向图的最小环问题。在本题中,你需要输出最小环的方案,若最小环不唯一,输出任意一个均可。若无解,输出 No solution.。图的节点数不超过 100 。
输入
第一行两个正整数 n,m 表示点数和边数。
接下来 m 行,每行三个正整数 x,y,z ,表示节点 x,y 之间有一条长度为 z 的边。
输出
输出一个最小环的方案:按环上顺序输出最小环上的点。若最小环不唯一,输出任意一个均可。若无解,输出 No solution.
样例输入
5 7
1 4 1
1 3 300
3 1 10
1 2 16
2 3 100
2 5 15
5 3 20
样例输出
标签
CEOI1999
题解
其实这道题可以用Floyd。
容易知道,树加上一条非树边可以形成环。利用这个性质,我们可以先求出一棵最小生成树,然后通过枚举每一条非树边求出加上它之后所形成的环的大小并尝试更新答案。输出方案时直接在最小生成树上暴力跳就行了。时间复杂度O(mlogn)。
代码:
#include<bits/stdc++.h>
using namespace std;
#define gc getchar
template<typename T>void read(T&cn){
char c;int sig=;
while(!isdigit(c=gc())) if(c=='-') sig=-;cn=c-;
while( isdigit(c=gc())) cn=cn*+c-;cn*=sig;
}
#define N 100010
#define INF 0x3f3f3f3f
int n,m;
namespace T{
int num,f[N];
struct node{
int u,v,w,nxt;
}e[N<<];
void add(int u,int v,int w){
e[++num]=(node){u,v,w,f[u]};f[u]=num;
e[++num]=(node){v,u,w,f[v]};f[v]=num;
}
//build graph
int dep[N],dis[N],faz[N],siz[N],son[N],top[N];
void dfs1(int u,int fa,int d){
dep[u]=d;faz[u]=fa;siz[u]=;
for(int i=f[u];i;i=e[i].nxt){
int v=e[i].v;
if(v==fa) continue;
dis[v]=dis[u]+e[i].w;dfs1(v,u,d+);siz[u]+=siz[v];
if(siz[v]>siz[son[u]]) son[u]=v;
}
}
void dfs2(int u,int st){
top[u]=st;
if(!son[u]) return;dfs2(son[u],st);
for(int i=f[u];i;i=e[i].nxt){
int v=e[i].v;
if(v==faz[u]||v==son[u]) continue;
dfs2(v,v);
}
}
int LCA(int x,int y){
int xx=top[x],yy=top[y];
while(xx^yy){
if(dep[xx]<dep[yy]){swap(xx,yy);swap(x,y);}
x=faz[xx];xx=top[x];
}
return dep[x]<dep[y]?x:y;
}
//tree dissection
}
namespace G{
struct node{
int u,v,w;
}e[N];
bool operator < (node a,node b){return a.w<b.w;}
//graph
int fa[N];
int find(int x){return x==fa[x]?x:fa[x]=find(fa[x]);}
int merge(int x,int y){
int xx=find(x),yy=find(y);
if(xx==yy) return ;
fa[xx]=yy; return ;
}
//Disjoint-Set
int vis[N];
void kruskal(){
sort(e+,e+m+);int cnt=;
for(int i=;i<=n;i++) fa[i]=i;
for(int i=;i<=m;i++){
int u=e[i].u,v=e[i].v;
if(merge(u,v)){cnt++;vis[i]=;T::add(u,v,e[i].w);}
if(cnt==n-) break;
}
}
//mst
int sum=INF,a,b,c;
#define pb push_back
vector<int>ans;
void work(){
read(n);read(m);
for(int i=;i<=m;i++){read(e[i].u);read(e[i].v);read(e[i].w);}
kruskal();T::dfs1(,,);T::dfs2(,);
for(int i=;i<=m;i++){
if(vis[i]) continue;
int u=e[i].u,v=e[i].v;
if(T::dep[u]<T::dep[v]) swap(u,v);
if(T::faz[u]==v||u==v) continue;
int lca=T::LCA(u,v),tmp=T::dis[u]+T::dis[v]-(T::dis[lca]<<)+e[i].w;
if(sum>tmp){sum=tmp;a=u,b=v;c=lca;}
}
if(sum==INF){puts("No solution.");exit();}
while(a^c){printf("%d ",a);a=T::faz[a];}printf("%d ",c);
while(b^c){ans.pb(b);b=T::faz[b];}
if(ans.size()) for(int i=ans.size()-;~i;i--) printf("%d ",ans[i]);
}
}
int main(){
G::work();
return ;
}
WOJ#3836 Sightseeing Trip的更多相关文章
- URAL 1004 Sightseeing Trip(最小环)
Sightseeing Trip Time limit: 0.5 secondMemory limit: 64 MB There is a travel agency in Adelton town ...
- poj1734 Sightseeing trip【最小环】
Sightseeing trip Time Limit: 1000MS Memory Limit: 65536K Total Submissions:8588 Accepted:3224 ...
- 「LOJ#10072」「一本通 3.2 例 1」Sightseeing Trip(无向图最小环问题)(Floyd
题目描述 原题来自:CEOI 1999 给定一张无向图,求图中一个至少包含 333 个点的环,环上的节点不重复,并且环上的边的长度之和最小.该问题称为无向图的最小环问题.在本题中,你需要输出最小环的方 ...
- poj 1734 Sightseeing trip判断最短长度的环
Sightseeing trip Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 5590 Accepted: 2151 ...
- 【poj1734】Sightseeing trip
Sightseeing trip Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 8520 Accepted: 3200 ...
- Ural 1004 Sightseeing Trip
Sightseeing Trip Time Limit: 2000ms Memory Limit: 16384KB This problem will be judged on Ural. Origi ...
- POJ 1734:Sightseeing trip
Sightseeing trip Time Limit: 1000MS Memory Limit: 65536K Total Submissions: Accepted: Special Judge ...
- [CEOI1999]Sightseeing trip(Floyed)
[CEOI1999]Sightseeing trip Description There is a travel agency in Adelton town on Zanzibar island. ...
- 「POJ1734」Sightseeing trip
「POJ1734」Sightseeing trip 传送门 这题就是要我们求一个最小环并且按顺序输出一组解. 考虑 \(O(n^3)\) 地用 \(\text{Floyd}\) 求最小环: 考虑 \( ...
随机推荐
- HBase过滤器(转载)
HBase为筛选数据提供了一组过滤器,通过这个过滤器可以在HBase中的数据的多个维度(行,列,数据版本)上进行对数据的筛选操作,也就是说过滤器最终能够筛选的数据能够细化到具体的一个存储单元格上(由行 ...
- 【leetcode】Decode Ways
题目如下: 解题思路:这个题目的本质是一个爬楼梯问题,在爬楼梯问题中,每次可以选择走一步或者两步.而本题也是一样的,解码的时候选择解码一个字符或者两个字符,但是加大了一点点难度,要考虑这些情况.1,Z ...
- React Native 中 跨页面间通信解决方案之 react-native-event-bus
https://github.com/crazycodeboy/react-native-event-bus 用法: A页面和B页面中都有相同的列表,点击B页面中的收藏按钮,A页面会跟着更新 impo ...
- 【CF1237D】Balanced Playlist(set,二分,线段树)
题意:给定一个n首歌的播放列表,第i首的值为a[i],听完第i首会回到第1首 现在从每首开始往下,记录听过的最大值,如果当前听的值严格小于听过最大值的一半则停止 问从每首歌开始往下听能听几首,不会停止 ...
- es之java搜索文档
1:搜索文档数据(单个索引) @Test public void getSingleDocument(){ GetResponse response = client.prepareGet(" ...
- 11 November
Weakness 求数列区间 \(\{a_n\}\) 中满足 \(i < j < k, a_i > a_j > a_k\) 的 \((i, j, k)\) 对的数目. 设对 \ ...
- [CSP-S模拟测试]:树(树上上升序列+主席树+线段树)
题目传送门(内部题78) 输入格式 第一行输入两个整数$n,q$,表示节点数和询问数. 第二行输入$n$个整数$w_i$,表示第$i$个点的智商. 第三行至第$n+1$行每行输入两个数$x,y$,表示 ...
- flex几种多列布局
基本的等分三列布局 .container{ display: flex; width: 500px; height: 200px; } .left{ flex:1; background: red; ...
- win10+VS2015+opencv3.4.0配置方法
win10+VS2015+opencv3.4.0配置方法 操作环境: windows10 64位opencv 3.4.0:https://opencv.org/releases.html(选择open ...
- Fabric基础架构原理(一)
Linux基金会于2015年12月启动了名为“超级账本”(Hyperledger)的开源项目,旨在推动各方协作,共同打造基于区块链的企业级分布式账本底层技术,用于构建支撑业务的行业应用和平台. 超级账 ...