hdu Minimum Transport Cost(按字典序输出路径)
http://acm.hdu.edu.cn/showproblem.php?
pid=1385
求最短路。要求输出字典序最小的路径。
spfa:拿一个pre[]记录前驱,不同的是在松弛的时候。要考虑和当前点的dis值相等的情况,解决的办法是dfs找出两条路径中字典序较小的。pre[]去更新。
把路径当做字符串处理。
我仅仅用之前的pre去更新当前点,并没考虑到起点到当前点的整个路径,事实上这样并不能保证是字典序最小。wa了N次。于是乎搜了下题解,发现用spfa解的非常少。看到了某大牛的解法如上,感觉非常赞的想法。
#include <stdio.h>
#include <algorithm>
#include <set>
#include <map>
#include <vector>
#include <math.h>
#include <string.h>
#include <queue>
#define LL long long
#define _LL __int64
using namespace std;
const int INF = 0x3f3f3f3f;
const int maxn = 110; int n;
int tax[maxn];
int Map[maxn][maxn];
int dis[maxn],inque[maxn];
int pre[maxn];
int pos = 0; void dfs(int u, char *s)
{
if(u == -1)
return;
dfs(pre[u],s);
s[pos++] = u+'0';
} bool solve(int v, int u)
{
char s1[maxn],s2[maxn]; //寻找先前的路径
pos = 0;
dfs(v,s1);
s1[pos] = '\0';
//寻找由u更新的路径
pos = 0;
dfs(u,s2);
s2[pos++] = v+'0';
s2[pos] = '\0'; if(strcmp(s1,s2) > 0) return true;
return false; } int spfa(int s, int t)
{
queue <int> que;
memset(dis,INF,sizeof(dis));
memset(inque,0,sizeof(inque));
memset(pre,-1,sizeof(pre)); dis[s] = 0;
inque[s] = 1;
que.push(s); while(!que.empty())
{
int u = que.front();
que.pop();
inque[u] = 0; for(int v = 1; v <= n; v++)
{
if(Map[u][v] > 0)
{
int tmp = dis[u] + Map[u][v] + tax[v]; if(tmp < dis[v])//直接更新
{
dis[v] = tmp;
pre[v] = u;
if(!inque[v])
{
inque[v] = 1;
que.push(v);
}
}
else if(tmp == dis[v] && solve(v,u))
{
pre[v] = u;
} }
}
} return dis[t];
} void output(int s, int t)
{
int res[maxn];
int cnt = 0;
int tmp = t; while(pre[tmp] != -1)
{
res[cnt++] = tmp;
tmp = pre[tmp];
}
res[cnt] = s;
printf("Path: ");
for(int i = cnt; i >= 1; i--)
printf("%d-->",res[i]);
printf("%d\n",res[0]);
} int main()
{
while(~scanf("%d",&n) && n)
{
for(int i = 1; i <= n; i++)
{
for(int j = 1; j <= n; j++)
scanf("%d",&Map[i][j]);
}
for(int i = 1; i <= n; i++)
scanf("%d",&tax[i]);
int u,v;
while(~scanf("%d %d",&u,&v))
{
if(u == -1 && v == -1) break;
int tmp1 = tax[u];//备份
int tmp2 = tax[v]; tax[u] = 0;
tax[v] = 0; printf("From %d to %d :\n",u,v);
int ans = spfa(u,v);
output(u,v);
printf("Total cost : %d\n\n",ans);
//恢复备份
tax[u] = tmp1;
tax[v] = tmp2; }
}
return 0;
}
floyd:pre[i][j]记录i到j路径上距离i近期的点,输出路径时按pre正向输出。感觉floyd非常强大啊,还能够这样记录路径。
#include <stdio.h>
#include <algorithm>
#include <set>
#include <map>
#include <vector>
#include <math.h>
#include <string.h>
#include <stack>
#include <queue>
#define LL long long
#define _LL __int64
using namespace std; const int INF = 0x3f3f3f3f;
const int maxn = 110;
int Map[maxn][maxn];
int tax[maxn];
int pre[maxn][maxn];
int n; void floyd()
{
//初始化
for(int i = 1; i <= n; i++)
for(int j = 1; j <= n; j++)
pre[i][j] = j; for(int k = 1; k <= n; k++)
{
for(int i = 1; i <= n; i++)
{
for(int j = 1; j <= n; j++)
{
if(Map[i][k] > 0 && Map[k][j] > 0)
{
int tmp = Map[i][k] + Map[k][j] + tax[k]; if(tmp < Map[i][j]) //小于直接更新
{
Map[i][j] = tmp;
pre[i][j] = pre[i][k];
}
else if(tmp == Map[i][j])
{
pre[i][j] = min(pre[i][k],pre[i][j]);
}
}
}
}
}
} void output(int s, int t)
{
printf("Path: ");
int tmp = s;
while(tmp != t)
{
printf("%d-->",tmp);
tmp = pre[tmp][t];
}
printf("%d\n",t); } int main()
{
while(~scanf("%d",&n) && n)
{
for(int i = 1; i <= n; i++)
for(int j = 1; j <= n; j++)
{
scanf("%d",&Map[i][j]);
if(Map[i][j] == -1)
Map[i][j] = INF;
} for(int i = 1; i <= n; i++)
scanf("%d",&tax[i]); int u,v;
floyd();
while(~scanf("%d %d",&u,&v))
{
if(u == -1 && v == -1) break;
printf("From %d to %d :\n",u,v);
output(u,v);
printf("Total cost : %d\n\n",Map[u][v]);
}
}
return 0;
}
hdu Minimum Transport Cost(按字典序输出路径)的更多相关文章
- HD1385Minimum Transport Cost(Floyd + 输出路径)
Minimum Transport Cost Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/O ...
- HDU 1385 Minimum Transport Cost( Floyd + 记录路径 )
链接:传送门 题意:有 n 个城市,从城市 i 到城市 j 需要话费 Aij ,当穿越城市 i 的时候还需要话费额外的 Bi ( 起点终点两个城市不算穿越 ),给出 n × n 大小的城市关系图,-1 ...
- ZOJ 1456 Minimum Transport Cost(floyd+后继路径记录)
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1456 题意:求最短路并且输出字典序最小的答案. 思路:如果用dijkstr ...
- hdu 1385 Minimum Transport Cost(floyd && 记录路径)
Minimum Transport Cost Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/O ...
- HDU 1385 Minimum Transport Cost (Dijstra 最短路)
Minimum Transport Cost http://acm.hdu.edu.cn/showproblem.php?pid=1385 Problem Description These are ...
- Minimum Transport Cost(floyd+二维数组记录路径)
Minimum Transport Cost Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/O ...
- hdu 1385 Minimum Transport Cost (Floyd)
Minimum Transport CostTime Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Ot ...
- HDU1385 Minimum Transport Cost (Floyd)
Minimum Transport Cost Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/O ...
- ZOJ 1456 Minimum Transport Cost(Floyd算法求解最短路径并输出最小字典序路径)
题目链接: https://vjudge.net/problem/ZOJ-1456 These are N cities in Spring country. Between each pair of ...
随机推荐
- 洛谷 P1400 塔
P1400 塔 题目描述 有N(2<=N<=600000)块砖,要搭一个N层的塔,要求:如果砖A在砖B上面,那么A不能比B的长度+D要长.问有几种方法,输出 答案 mod 10000000 ...
- JavaScript作用域闭包(你不知道的JavaScript)
JavaScript闭包.是JS开发project师必须深入了解的知识. 3月份自己曾撰写博客<JavaScript闭包>.博客中仅仅是简单阐述了闭包的工作过程和列举了几个演示样例,并没有 ...
- 从头认识Spring-2.3 注解装配-@autowired(4)-required(1)
这一章节我们来具体讨论一下@autowired里面的參数required. 1.domain(重点) 蛋糕类: package com.raylee.my_new_spring.my_new_spri ...
- 将bat批处理命令文件固定到任务栏
将bat批处理命令文件固定到任务栏第一种方法:使用链接工具http://www.xstui.com/read/3451.在任务栏点击右键,移动到工具栏,勾选链接工具2.你会在通知栏左侧看到链接字样,将 ...
- [NOI.AC#41]最短路 线性基
链接 题解 如果不加边,两个点之间的长度是唯一的(只能走最短路径),因为如果重复走,就异或掉了. 因此,先DFS预处理一下每个点到根的距离 \(d[x]\) ,那么 \(x,y\) 之间的距离为 $d ...
- 17.Node.js 回调函数--异步编程
转自:http://www.runoob.com/nodejs/nodejs-tutorial.html Node.js 异步编程的直接体现就是回调. 异步编程依托于回调来实现,但不能说使用了回调后程 ...
- Android视频播放软解与硬解的区别
硬解,用自带播放器播放,android中的VideoView 软解,使用音视频解码库,比如FFmpeg 一.硬解码 硬解:就是调用GPU的专门模块编码来解,减少CPU运算,对CPU等硬件要求也相对低点 ...
- BZOJ1009: [HNOI2008]GT考试(KMP+矩阵乘法)
Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字.他的不吉利数学A1A2...Am(0< ...
- APK文件浅析-Android
2011~2015,5年时间,断断续续学习了Android. 最近打算在2011年2个月认真学习的基础上,深入学习下. 由于有之前的Android基础,加上N年的Java等变成经验,自我感觉And ...
- Css 显示删除条目效果
样式设置