Travel

Description

  给出一个有 个顶点 条边的有向图,对于一条边长度为len的边有两种走法。
  1、如果a和b可以互达,则走过这条边的时间为len
  2、如果a和b不可以互达,则走过这条边的时间为2*len
  现在给出一个k,问,从顶点1到顶点n,满足第二种走法不超过k次的最短时间是多少。

Input

  第一行有3个整数n,m,k(1<=n<=100,1<=m<=10000,0<=k<=10),表示有n个顶点,m条边。
  接下来有m行,每行有3个整数xi,yi,leni(1<=xi,yi<=n,1<=leni<=10000),表示长度为leni的有向边。
  注意,两个点可能有多条边连接。

Output

  一行一个整数,表示最短时间。
  如果没有满足题目条件的路径,则输出-1

Sample Input

7 7 3

1 2 2

1 3 2

2 4 3

4 7 5

3 5 4

5 6 1

6 4 2

Sample Output

20

题目解释

这道题我在做的时候被坑了一下,就是在:
这里写图片描述
这段描述上,我理解成了如果a可以到b,那么长度为len,而如果在这时b不可以到a,那么可以使长度为2*len后,也可以通过。
而实际上,题目的意思是有n个点,m条有向边,其中任意两个点a,b,它们间的距离分别为 lena lenb ,如果a可以到b,b也可以到a,那么它们之间的距离不变,但如果a可以到b,b却不可以到a,那么a到b的距离为 2lena ,反之相同。
那么在走这种不可互达的边小于等于k次时,最少的距离是多少?

题解

先用Floyd求出点与点之间的最短路,再用分层SPFA求出走不可互达边的最小距离。
因为对于这种算法我也很难再多加解释,所以不理解的还是看下面吧。


var
    x,y,s,n,m,k,i,j,o,ans:longint;
    a,b:array[0..1000,0..1000] of longint;
    f:array[1..1000,0..100] of longint;
    d:array[1..1000,1..2] of longint;
    bz:array[1..1000,0..100] of boolean;
procedure spfa;
var
    l,r,i,j,k,o:longint;
begin
    fillchar(d,sizeof(d),0);
    fillchar(bz,sizeof(bz),false);
    fillchar(f,sizeof(f),$7f div 3);
    f[1,0]:=0;
    bz[1,0]:=true;
    d[1,1]:=1;
    d[1,2]:=0;
    l:=1;r:=1;
    while l<=r do
    begin
        k:=d[l,1];
        j:=d[l,2];
        for i:=1 to n do if a[k,i]<>a[0,0] then
        begin
            if b[i,k]=1 then o:=0 else o:=1;
            if f[i,j+o]>a[k,i]+f[k,j] then
            begin
                f[i,j+o]:=a[k,i]+f[k,j];
                if not bz[i,j+o] then
                begin
                    inc(r);
                    d[r,1]:=i;
                    d[r,2]:=j+o;
                    bz[i,j+o]:=true;
                end;
            end;
        end;
        bz[k,j]:=false;
        inc(l);
    end;
end;
begin
    readln(n,m,k);
    fillchar(a,sizeof(a),$7f div 3);
    for i:=1 to n do
        for j:=1 to n do b[i,j]:=2;
    for i:=1 to m do
    begin
        readln(x,y,s);
        b[x,y]:=1;
        if a[x,y]>s then a[x,y]:=s;
    end;
    for i:=1 to n do
        for j:=1 to n do
            for o:=1 to n do
                if (j<>o) and (i<>o) and (i<>j) then
                    if (b[i,o]=1) and (b[o,j]=1) then b[i,j]:=1;
    for i:=1 to n do
        for j:=1 to n do
            if a[i,j]<>a[0,0] then a[i,j]:=a[i,j]*b[j,i];
    spfa;
    ans:=a[0,0];
    for i:=0 to k do if ans>f[n,i] then ans:=f[n,i];
    if ans=a[0,0] then writeln(-1) else writeln(ans);
end.
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值