ACM/ICPC 之 Floyd+记录路径后继(Hdu1385-ZOJ1456)

需要处理好字典序最小的路径



HDU1385(ZOJ1456)-Minimum Transport

//Hdu1385-ZOJ1456
//给定邻接矩阵,求给定起点到终点的最短路径,若有相同路长的路径按照字典序输出
//Floyd比较适合此题
//网上看到的两种做法比较推荐
//第一种是Floyd+记录起点后继
//第二种是Floyd+深搜(从起点开始深搜至终点)-利用Floyd得到的最短路剪枝
//其他的解法则是按照SPFA,dijkstra解,当路长相同时需要进行比较(较繁琐)
//Time:0Ms	Memory:404K
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;

#define INF 0x3f3f3f3f
#define MAX 105

int n;
int	d[MAX][MAX];
int board[MAX][MAX], c[MAX];
int path[MAX][MAX];	//记录i到j最短路径中i的后继

void floyd()
{
	memcpy(d, board, sizeof(d));
	for (int k = 1; k <= n; k++)
		for (int i = 1; i <= n; i++)
			for (int j = 1; j <= n; j++)
			{
				if (d[i][j] > d[i][k] + d[k][j] + c[k])
				{
					d[i][j] = d[i][k] + d[k][j] + c[k];
					path[i][j] = path[i][k];
				}
				//相同路径下选择后继更小的
				else if (d[i][j] == d[i][k] + d[k][j] + c[k] && path[i][j] > path[i][k])
					path[i][j] = path[i][k];
			}
}

void output(int s,int e)
{
	printf("-->%d", path[s][e]);
	if (path[s][e] != e)
		output(path[s][e], e);
}

int main()
{
	while (scanf("%d", &n), n)
	{
		for (int i = 1; i <= n; i++)
			for (int j = 1; j <= n; j++)
			{
				scanf("%d", &board[i][j]);
				if (board[i][j] == -1) board[i][j] = INF;
				path[i][j] = j;
			}
		for (int i = 1; i <= n; i++)
			scanf("%d", &c[i]);
		floyd();
		int s, e;
		while (scanf("%d%d", &s, &e), s != -1 && e != -1)
		{
			printf("From %d to %d :\n", s, e);
			printf("Path: %d", s);
			if(s != e)	output(s, e);	//起点与终点不同开始递归
			printf("\nTotal cost : %d\n\n", d[s][e]);
		}
	}
	return 0;
}
时间: 2024-10-16 05:05:14

ACM/ICPC 之 Floyd+记录路径后继(Hdu1385-ZOJ1456)的相关文章

HDU 5137 How Many Maos Does the Guanxi Worth(floyd记录路径)

题意:给定N个点和M条边,点编号是1到N.现在要从2到N-1中选择一个删除,同时跟选择的点连接的边也就消失,然后使得点1到N的最短路径的长度最大.如果点1和点N不连通,则输出"Inf". 思路:直接暴力,枚举删去的点即可.我做了一步优化,只删原图最短路上的点, 所以用floyd的时候记录路径即可 //Accepted 1164 KB 0 ms #include<cstdio> #include<iostream> #include<cstring>

ACM/ICPC 之 Floyd练习六道(ZOJ2027-POJ2253-POJ2472-POJ1125-POJ1603-POJ2607)

以Floyd解法为主的练习题六道 ZOJ2027-Travelling Fee //可免去一条线路中直接连接两城市的最大旅行费用,求最小总旅行费用 //Time:0Ms Memory:604K #include<iostream> #include<cstring> #include<cstdio> #include<algorithm> using namespace std; #define MAX 205 #define MAXS 12 #define

POJ 1734 Sightseeing trip (Floyd 最小环+记录路径)

Sightseeing trip Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5040   Accepted: 1932   Special Judge Description There is a travel agency in Adelton town on Zanzibar island. It has decided to offer its clients, besides many other attra

[Python] 弗洛伊德(Floyd)算法求图的直径并记录路径

相关概念 对于一个图G=(V, E),求图中两点u, v间最短路径长度,称为图的最短路径问题.最短路径中最长的称为图的直径. 其中,求图中确定的某两点的最短路径算法,称为单源最短路径算法.求图中任意两点间的最短路径算法,称为多源最短路径算法. 常用的路径算法有: Dijkstra算法 SPFA算法\Bellman-Ford算法 Floyd算法\Floyd-Warshall算法 Johnson算法 其中最经典的是Dijkstra算法和Floyd算法.Floyd算法是多源最短路径算法,可以直接求出图

ACM/ICPC 之 经典动规(POJ1088-滑雪)

POJ1088-滑雪 将每个滑雪点都看作起点,从最低点开始逐个由四周递推出到达此点的最长路径的长度,由该点记下. 理论上,也可以将每一点都看作终点,由最高点开始计数,有兴趣可以试试. 1 //经典DP-由高向低海拔滑雪-求最长路 2 //Memory:372K Time:32 Ms 3 #include<iostream> 4 #include<cstring> 5 #include<cstdio> 6 #include<algorithm> 7 using

《ACM/ICPC 算法训练教程》读书笔记一之数据结构(堆)

书籍简评:<ACM/ICPC 算法训练教程>这本书是余立功主编的,代码来自南京理工大学ACM集训队代码库,所以小编看过之后发现确实很实用,适合集训的时候刷题啊~~,当时是听了集训队final的意见买的,感觉还是不错滴. 相对于其他ACM书籍来说,当然如书名所言,这是一本算法训练书,有着大量的算法实战题目和代码,尽管小编还是发现了些许错误= =,有部分注释的语序习惯也有点不太合我的胃口.实战题目较多是比较水的题,但也正因此才能帮助不少新手入门,个人认为还是一本不错的算法书,当然自学还是需要下不少

hdu 1226 BFS + bfs记录路径

http://acm.hdu.edu.cn/showproblem.php?pid=1226 为了省空间,可以用vis数组初始化的时候初始化为-1, 发现一个BFS容易错的地方 开始一直WA在这里:就是我int tp=q.front();之后马上q.pop():了,然后才去判断是不是符合条件以break,这样就不能根据q.empty()==1认为没有找到ans 因为这里WA了 其实也可以vis[0] == -1来判断 比较不理解的是 当n==0的时候 %n==0的时候怎么处理 //#pragma

ACM ICPC 2008–2009 NEERC MSC A, B, C, G, L

这套题是我上周日, 就是前天打得一场组队赛, 题目不太好找 题目链接:http://codeforces.com/gym/100861 在virtual judge 上也可以提交哦! A ACM ICPC Rules: 题目大意: 有很多所高校参加预选赛, 并在预选赛取得了排名, 但是对于每所学校, 除了MSU有4个名额之外其他大学只有两个名额( 也就是说, 只有每个大学的前2名进决赛(MSU前四名)&& 最多有10个队伍进入决赛), 高中队伍不能进入决赛. 给出预选赛的排名, 输出可以进

2014 ACM/ICPC Asia Regional Xi&#39;an Online

03 hdu5009 状态转移方程很好想,dp[i] = min(dp[j]+o[j~i]^2,dp[i]) ,o[j~i]表示从j到i颜色的种数. 普通的O(n*n)是会超时的,可以想到o[]最大为sqrt(n),问题是怎么快速找到从i开始往前2种颜色.三种.四种...o[]种的位置. 离散化之后,可以边走边记录某个数最后一个出现的位置,初始为-1,而所要求的位置就等于 if(last[a[i]]==-1) 该数没有出现过,num[i][1] = i,num[i][j+1] = num[i-1