想看更多的解题报告: http://iyenn.com/rec/1824570.html
转载请注明出处:http://blog.csdn.net/wangjian8006
题目大意:给出你一个无向图,要你求一个最小环,最小环的最少有3个节点,如果有环则顺序输出节点上的点,如果没有环,输出NO solution.
数据比较弱,测试数据是有重边的,但是我没考虑都过了.
解题思路:刚看到这题有点凌乱,没思路,后来想到深搜去找一个最小环居然过了
前面我的想法是:一个最小环,对于环上的两点,环的路径是等于两点的最短路径与次短路径只和的,不过接下来就断了。然后直接暴力深搜写了
在网上找了资料,发现是用floyd变形去做,跟着模板又写了一次,看懂了感觉好神奇
- /*
- DFS
- Memory 212K
- Time 125MS
- */
- #include
- #include
- using namespace std;
- #define MAXV 110
- #define INF INT_MAX
-
- int map[MAXV][MAXV];
- bool mark[MAXV];
- int ans[MAXV],ans_sum;
- int n,m,key,head;
- int temp[MAXV];
-
- void dfs(int x,int sum,int ns,int parent){ //当前节点,从head开始找到x的路径长度,找到了多少个节点,x的父亲节点
- int i;
- if(x==head && mark[head]){
- if(key>sum){
- key=sum;
- ans_sum=ns;
- for(i=0;i
- ans[i]=temp[i];
- }
- return ;
- }
- temp[ns]=x;
- for(i=1;i<=n;i++){
- if(!map[x][i]) continue;
- if(i!=head && mark[i]) continue; //如果t被找过,而且t不是head的环,那么不用找
- if(i==parent) continue; //不能往回走
- mark[i]=1;
- if(sum+map[x][i]
dfs(i,sum+map[x][i],ns+1,x); - mark[i]=0;
- }
- }
-
- int main(){
- int i,a,b,c;
- while(~scanf("%d%d",&n,&m)){
- memset(map,0,sizeof(map));
- for(i=0;i
- scanf("%d%d%d",&a,&b,&c);
- map[a][b]=map[b][a]=c;
- }
- key=INF;
- for(i=1;i<=n;i++){ //对于每个节点,找出这节点的最小环
- memset(mark,0,sizeof(mark));
- head=i;
- dfs(i,0,0,0);
- }
- if(key==INF){
- printf("No solution.\n");
- }else{
- for(i=0;i
printf("%d ",ans[i]); - printf("\n");
- }
- }
-
- return 0;
- }
=========================================================================================================
- /*
- floyd
- Memory 284K
- Time 16MS
- */
- #include
- using namespace std;
- #define MAXV 101
- #define INF 0x7ffffff
- #define min(a,b) a
-
- int map[MAXV][MAXV],dist[MAXV][MAXV],pre[MAXV][MAXV],ans[MAXV],n;
- int key,ans_sum;
-
- void floyd(){
- int i,j,k;
- key=INF;
- for(k=1;k<=n;k++){
- //最短路径外一点将最短路首尾链接,那么就得到一个最小环
- for(i=1;i
- for(j=i+1;j
- //求最小环不能用两点间最短路松弛,因为(i,k)之间的最短路,(k,j)之间的最短路可能有重合的部分
- //所以map[][]其实是不更新的,这里和单纯的floyd最短路不一样
- //dist[i][j]保存的是 i 到 j 的最短路权值和
- int tmp=dist[i][j]+map[i][k]+map[k][j];//这里 k 分别和 i 还有 j 在mat中直接相连
- if(tmp
- key=tmp;
- ans_sum=0;
- int p=j;
- while(p!=i){//回溯
- ans[ans_sum++]=p;
- p=pre[i][p];//pre[i][j]表示 i 到 j 最短路径上 j 前面的一个点
- }
- ans[ans_sum++]=i;
- ans[ans_sum++]=k;
- }
- }
- }
- for(i=1;i<=n;i++){
- for(j=1;j<=n;j++){
- if(dist[i][j]>dist[i][k]+dist[k][j]){
- dist[i][j]=dist[i][k]+dist[k][j]; //dist[][]保存两点间最短距离
- pre[i][j]=pre[k][j];
- }
- }
- }
- }
- }
-
- int main(){
- int i,j,m,a,b,c;
-
- while(~scanf("%d%d",&n,&m)){
- for(i=1;i<=n;i++){
- for(j=1;j<=n;j++){
- map[i][j]=dist[i][j]=INF;
- pre[i][j]=i;
- }
- }
- while(m--){
- scanf("%d%d%d",&a,&b,&c);
- map[a][b]=map[b][a]=dist[a][b]=dist[b][a]=min(map[a][b],c);
- }
-
- floyd();
-
- if(key==INF)puts("No solution.\n");
- else{
- printf("%d",ans[0]);
- for(i=1;i
- printf(" %d",ans[i]);
- printf("\n");
- }
- }
- return 0;
- }
注:本文转载自blog.csdn.net的wangjian8006的文章"http://blog.csdn.net/wangjian8006/article/details/7979960"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
评论记录:
回复评论: