6.6模拟题 2009年汕头市信息学奥林匹克竞赛 过路费1790

题目

古时候有N个城市,编号分为1到N,每两个城市之间有且只有一条路。商人在城市之前做买卖时就会遇到一个问题,那就是每通过一条路,就要支付一定的过路费,这个过路费等于商人身上的金钱数乘以一个不大于1的小数,并且这个小数会因道路的不同而不同。商人想从城市A去到城市B,问最多可以剩下多少钱。
n<=500

题解

很显然,这是一道单源最短路!
Floyd当然简单易行,但是多半会爆几个点
因此我们用dij(这个单词全称不记得了)
此题注意要求留下的路费最多,因此”>”“<”要注意用对
Dij过程:
d[i]放从起点s到所有点的路程。
a[u,v]放从u到v的路程
b[i]放从起点s到i点的最短路是否求过
a)初始化:
d和a全都弄成很大的数
for i 1——n d[i]=a[s,i]//所有可以从1直接到的点

b)repeat

找一个没有求过最短路且距起点最近的点u
若存在u则用d[u]去更新别的点
i=1——n
   d[i]=d[u]+a[u,i](d[i]>d[u]+a[u,i]&i点没有求完最短路)
b[u]标记为求过

until u=0

O(n2)

代码

var
  n,x,y,i,j:longint;
  a:array[0..500,0..500]of real;
  d:array[1..500]of real;
  c:real;

procedure dij;
var
  i,u:longint;
  min:real;
  v:array[1..500]of boolean;
begin
  for i:=1 to n do d[i]:=a[x,i];
  fillchar(v,sizeof(v),true);v[x]:=false;
  repeat
    min:=0;u:=0;
    for i:=1 to n do
      if (d[i]>min)and v[i] then
        begin min:=d[i];u:=i;end;
    if u>0 then
      begin
        for i:=1 to n do
          if d[u]*a[u,i]>d[i] then
            d[i]:=d[u]*a[u,i];
          v[u]:=false;
      end;
  until u=0;
end;

begin
  assign(input,'rate.in');
  assign(output,'rate.out');
  reset(input);rewrite(output);
  readln(n,x,y);
  fillchar(a,sizeof(a),$7);
  for i:=1 to n do
    begin
      for j:=1 to n do
        begin
          read(c);
          a[i,j]:=1-c;
        end;
      a[i,i]:=a[0,0];
    end;
  dij;
  writeln(d[y]:0:2);
  close(input);close(output);
end.


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值