题目描述:

在每年的校赛里,所有进入决赛的同学都会获得一件很漂亮的t-shirt。但是每当我们的工作人员把上百件的衣服从商店运回到赛场的时候,却是非常累的!所以现在他们想要寻找最短的从商店到赛场的路线,你可以帮助他们吗?

输入:

输入包括多组数据。每组数据第一行是两个整数N、M(N<=100,M<=10000),N表示成都的大街上有几个路口,标号为1的路口是商店所在地,标号为N的路口是赛场所在地,M则表示在成都有几条路。N=M=0表示输入结束。接下来M行,每行包括3个整数A,B,C(1<=A,B<=N,1<=C<=1000),表示在路口A与路口B之间有一条路,我们的工作人员需要C分钟的时间走过这条路。输入保证至少存在1条商店到赛场的路线。
当输入为两个0时,输入结束。

输出:

对于每组输入,输出一行,表示工作人员从商店走到赛场的最短时间。

样例输入:
2 1
1 2 3
3 3
1 2 5
2 3 5
3 1 2
0 0
样例输出:
3
2
本来是一道很简单的迪杰特斯拉算法,一开始提交的代码如下
 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <cmath>
#define MAX 103
#define inf 0x3f3f3f3f
int map[MAX][MAX];
bool flag[MAX]; int main(int argc, char const *argv[])
{
int n, m;
scanf("%d %d",&n,&m);
while(n != || m != ) {
for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
map[i][j] = inf;
flag[i] = false;
}
} int a,b,c;
for(int i = ; i < m; i++) {
scanf("%d %d %d",&a,&b,&c);
map[a][b] = c;
map[b][a] = c;
} flag[] = true;
int count = ;
while(count <= n) {
int min = MAX;
int mini = -;
for(int i = ; i <= n; i++) {
if(flag == false) {
if(map[][i] < min) {
min = map[][i];
mini = i;
}
}
}
map[][mini] = min;
flag[mini] = true;
for(int i = ; i <= n; i++) {
int dis = map[][mini] + map[mini][i];
if(dis < map[][i]) {
map[][i] = dis;
}
}
count++;
}
printf("%d\n",map[][n]);
scanf("%d %d",&n,&m);
}
return ;
}

一开始百思不得其解究竟哪里做错了,只好先换一种写法

 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <cmath>
#define MAX 103
#define inf 999999999
int map[MAX][MAX];
bool flag[MAX];
int lowCost[MAX]; int main(int argc, char const *argv[])
{
int n, m;
scanf("%d %d",&n,&m);
while(n != || m != ) {
for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
map[i][j] = inf;
}
flag[i] = false;
} int a,b,c;
for(int i = ; i < m; i++) {
scanf("%d %d %d",&a,&b,&c);
map[a][b] = c;
map[b][a] = c;
} flag[] = true;
for(int i = ; i <= n; i++) {
lowCost[i] = map[][i];
} for(int i = ; i <= n; i++) {
int min = inf;
int v = -;
for(int j = ; j <= n; j++) {
if(flag[j] == false && lowCost[j] < min) {
min = lowCost[j];
v = j;
}
}
flag[v] = true; for(int j = ; j <= n; j++) {
if(flag[j] == false && lowCost[v]+map[v][j] < lowCost[j]) {
lowCost[j] = lowCost[v]+map[v][j];
}
}
} printf("%d\n",lowCost[n]);
scanf("%d %d",&n,&m);
}
return ;
}

后来经仔细排查,第一种写法的错误有两处,一是33行,min = inf

二是36行flag[i] ==false

另一种办法是使用Floyd算法,但开始我的代码是这样的

 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <cmath>
#define MAX 103
#define inf 999999999
int map[MAX][MAX]; int main(int argc, char const *argv[])
{
int n, m;
scanf("%d %d",&n,&m);
while(n != || m != ) {
for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
map[i][j] = inf;
}
} for (int i=; i<=n; ++i)
map[i][i] = ; int a,b,c;
for(int i = ; i < m; i++) {
scanf("%d %d %d",&a,&b,&c);
map[a][b] = c;
map[b][a] = c;
} for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
for(int k = ; k <= n; k++) {
if (map[i][k] == inf || map[k][j] == inf)
continue;
if(map[i][k]+map[k][j] < map[i][j]) {
map[i][j] = map[i][k] + map[k][j];
}
}
}
}
printf("%d\n",map[][n]);
scanf("%d %d",&n,&m);
}
return ;
}

但WA了,而真正的算法是这样的

 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <cmath>
#define MAX 103
#define inf 999999999
int map[MAX][MAX]; int main(int argc, char const *argv[])
{
int n, m;
scanf("%d %d",&n,&m);
while(n != || m != ) {
for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
map[i][j] = inf;
}
} int a,b,c;
for(int i = ; i < m; i++) {
scanf("%d %d %d",&a,&b,&c);
map[a][b] = c;
map[b][a] = c;
} for(int k = ; k <= n; k++) {
for(int i = ; i <= n; i++) {
for(int j = ; j <= n; j++) {
if (map[i][k] == inf || map[k][j] == inf)
continue;
if(i != j && map[i][k] + map[k][j] < map[i][j]) {
map[i][j] = map[i][k] + map[k][j];
}
}
}
}
printf("%d\n",map[][n]);
scanf("%d %d",&n,&m);
}
return ;
}

注意算法是 先对中间点进行遍历,不断增加新的中间点以得到更优的答案

九度oj 题目1447:最短路的更多相关文章

  1. 九度OJ 题目1384:二维数组中的查找

    /********************************* * 日期:2013-10-11 * 作者:SJF0115 * 题号: 九度OJ 题目1384:二维数组中的查找 * 来源:http ...

  2. hdu 1284 关于钱币兑换的一系列问题 九度oj 题目1408:吃豆机器人

    钱币兑换问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  3. 九度oj题目&amp;吉大考研11年机试题全解

    九度oj题目(吉大考研11年机试题全解) 吉大考研机试2011年题目: 题目一(jobdu1105:字符串的反码).    http://ac.jobdu.com/problem.php?pid=11 ...

  4. 九度oj 题目1007:奥运排序问题

    九度oj 题目1007:奥运排序问题   恢复 题目描述: 按要求,给国家进行排名. 输入:                        有多组数据. 第一行给出国家数N,要求排名的国家数M,国家号 ...

  5. 九度oj 题目1087:约数的个数

    题目链接:http://ac.jobdu.com/problem.php?pid=1087 题目描述: 输入n个整数,依次输出每个数的约数的个数 输入: 输入的第一行为N,即数组的个数(N<=1 ...

  6. 九度OJ题目1105:字符串的反码

    tips:scanf,cin输入字符串遇到空格就停止,所以想输入一行字符并保留最后的"\0"还是用gets()函数比较好,九度OJ真操蛋,true?没有这个关键字,还是用1吧,还是 ...

  7. 九度oj题目1009:二叉搜索树

    题目描述: 判断两序列是否为同一二叉搜索树序列 输入:                        开始一个数n,(1<=n<=20) 表示有n个需要判断,n= 0 的时候输入结束. 接 ...

  8. 九度oj题目1002:Grading

    //不是说C语言就是C++的子集么,为毛printf在九度OJ上不能通过编译,abs还不支持参数为整型的abs()重载 //C++比较正确的做法是#include<cmath.h>,cou ...

  9. 九度OJ题目1003:A+B

    while(cin>>str1>>str2)就行了,多简单,不得不吐槽,九度的OJ真奇葩 题目描述: 给定两个整数A和B,其表示形式是:从个位开始,每三位数用逗号", ...

随机推荐

  1. Apache is running a threaded MPM, but your PHP module is not compiled to be threadsafe. you need to recompile php. pre-configuration failed

    手动配置想要组合版本的wamp环境时,在服务器上直接下载的几个安装包怎么都组合安装不成功,纠结很久,终于找到原因.配置apache支持php后apache一直无法成功启动.后来发现php是nts的版本 ...

  2. P1216 [USACO1.5]数字三角形 Number Triangles

    题目描述 观察下面的数字金字塔. 写一个程序来查找从最高点到底部任意处结束的路径,使路径经过数字的和最大.每一步可以走到左下方的点也可以到达右下方的点. 7 3 8 8 1 0 2 7 4 4 4 5 ...

  3. POJ 3461 kmp

    Oulipo Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 40168   Accepted: 16135 Descript ...

  4. 2840 WIKIOI——评测

    2840 WIKIOI——评测 时间限制: 1 s 空间限制: 2000 KB 题目等级 : 白银 Silver       题目描述 Description Wikioi上有一题有N个测试点,时限为 ...

  5. web前端性能优化 (share)

    本文转自:http://www.cnblogs.com/50614090/archive/2011/08/19/2145620.html 一. WEB前台的优化规则 一.尽量减少 HTTP 请求 有几 ...

  6. C# 语句 分支语句

    语句是指程序命令,按照顺序执行.可以分为   顺序语句  分支语句  循环语句 之前学习的内容都是按照顺序程序执行的,称之为顺序语句. 今天学的的内容是分支语句. 语句可以嵌套,可以是以分号结尾的单行 ...

  7. ios UI自动化测试

    转载:http://www.cnblogs.com/dokaygang128/p/3517674.html 一.一些注意事项: 1.做自动化测试时注意如果是真机话首先要设置不锁屏. 2.自动化测试过程 ...

  8. 复合词UVa10391(STL简单应用)

    一.题目 输入一系列由小写字母组成的单词.输入已按照字典序排序(这句话就是个陷阱),且不超过120000个.找出所有的复合词,即恰好由两个单词连接而成的单词. 二.解题思路 要么枚举两两拼接的情况,O ...

  9. PyTorch在NLP任务中使用预训练词向量

    在使用pytorch或tensorflow等神经网络框架进行nlp任务的处理时,可以通过对应的Embedding层做词向量的处理,更多的时候,使用预训练好的词向量会带来更优的性能.下面分别介绍使用ge ...

  10. python基础一 day11 装饰器(1)

    接收的时候是聚合,调用的时候是打散     print(*args)本来在里面用的时候是用args,是一个元祖,加上一个 * 号,把元祖解包了(打散了). from functools import ...