首先是稀疏图,不难想到dij+heap

观察题目可以知道,0<=k<=10;

所以比较裸的想法就是,d[i,j]表示已经免费了i条线路后到达j的最短路

容易得到

d[i,j]:=min(d[i,j],d[i-1,k]);

d[i,j]:=min(d[i,j],d[i,k]+w[k,j]);

然后在做dij在选择中间点的时候,我们穷举一下免费的线路数就可以了;

然后就要维护10个堆……

语言表述有点渣,具体还是看程序吧

 const inf=;
type point=record
       num,dis:longint;
     end;
     node=record
       next,cost,point:longint;
     end; var edge:array[..] of node;
    heap:array[..,..] of point;
    d,where:array[..,..] of longint;
    h:array[..] of longint;
    p:array[..] of longint;
    len,x,y,z,i,j,n,m,u,k,s,t:longint; procedure swap(var a,b:point);
  var c:point;
  begin
    c:=a;
    a:=b;
    b:=c;
  end; function min(a,b:longint):longint;
  begin
    if a>b then exit(b) else exit(a);
  end; procedure add(x,y,z:longint);
  begin
    inc(len);
    edge[len].point:=y;
    edge[len].cost:=z;
    edge[len].next:=p[x];
    p[x]:=len;
  end; procedure sift(p,i:longint);
  var j,x,y:longint;
  begin
    j:=i shl ;
    while j<=h[p] do
    begin
      if (j+<=h[p]) and (heap[p,j].dis>heap[p,j+].dis) then inc(j);
      if heap[p,i].dis>heap[p,j].dis then
      begin
        x:=heap[p,i].num;
        y:=heap[p,j].num;
        where[p,x]:=j;
        where[p,y]:=i;
        swap(heap[p,i],heap[p,j]);
        i:=j;
        j:=i shl ;
      end
      else break;
    end;
  end; procedure up(p,i:longint);
  var j,x,y:longint;
  begin
    j:=i shr ;
    while j> do
    begin
      if heap[p,i].dis<heap[p,j].dis then
      begin
        x:=heap[p,i].num;
        y:=heap[p,j].num;
        where[p,x]:=j;
        where[p,y]:=i;
        swap(heap[p,i],heap[p,j]);
        i:=j;
        j:=j shr ;
      end
      else break;
    end;
  end; begin
  len:=-;
  fillchar(p,sizeof(p),);
  readln(n,m,k);
  readln(s,t);
  inc(s);  //点序号都+,习惯一点
  inc(t);
  for i:= to m do
  begin
    readln(x,y,z);
    inc(x);
    inc(y);
    add(x,y,z);
    add(y,x,z);
  end;
  for i:= to n do
    if i<>s then
    begin
      for j:= to k do
        d[j,i]:=inf;
    end
    else
      for j:= to k do
        d[j,i]:=;
  for i:= to k do
  begin
    for j:= to n do
    begin
      heap[i,j].num:=j;
      heap[i,j].dis:=d[i,j];
      where[i,j]:=j;
    end;
    h[i]:=n;
  end;
  for i:= to k do
    up(i,s);  //起点不一定是0,要初始化堆,一开始这里被坑翻了
  for u:= to n do
  begin
    for j:=k downto do 
    begin
      x:=heap[j,].num;
      y:=heap[j,h[j]].num;  
      if (h[j]=) or (d[j,x]=inf) then continue;
      where[j,y]:=;  //dij+heap比较重要的细节
      swap(heap[j,],heap[j,h[j]]);
      dec(h[j]); 
      sift(j,);  //出堆,调整堆
      i:=p[x];
      while i<>- do
      begin
        y:=edge[i].point;
        if (j<>k) and (d[j+,y]>d[j,x]) then   //两种情况
        begin
          d[j+,y]:=d[j,x];
          heap[j+,where[j+,y]].dis:=d[j+,y];
          up(j+,where[j+,y]);
        end;
        if (d[j,y]>d[j,x]+edge[i].cost) then
        begin
          d[j,y]:=d[j,x]+edge[i].cost;
          heap[j,where[j,y]].dis:=d[j,y];
          up(j,where[j,y]);
        end;
        i:=edge[i].next;
      end;
    end;
  end;
  writeln(d[k,t]); //肯定尽可能免费好啊
end.

bzoj2763的更多相关文章

  1. BZOJ2763[JLOI2011]飞行路线 [分层图最短路]

    2763: [JLOI2011]飞行路线 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2523  Solved: 946[Submit][Statu ...

  2. 【BZOJ2763】飞行路线(最短路)

    [BZOJ2763]飞行路线(最短路) 题面 BZOJ Description Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司.该航空公司一共在n个城市设有业务,设这些城市分别标 ...

  3. BZOJ2763 JLOI2011 飞行路线 【最短路+DP】

    BZOJ2763 JLOI2011 飞行路线 Description Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司.该航空公司一共在n个城市设有业务,设这些城市分别标记为0到n ...

  4. 【bzoj2763】[JLOI2011]飞行路线 (分层图最短路)(优先队列dij)

    [bzoj2763][JLOI2011]飞行路线 2014年3月25日1,7260 Description Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司.该航空公司一共在n个城 ...

  5. 分层图最短路【bzoj2763】: [JLOI2011]飞行路线

    bzoj2763: [JLOI2011]飞行路线 Description Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司.该航空公司一共在n个城市设有业务,设这些城市分别标记为0 ...

  6. BZOJ2763 [JLOI2011]飞行路线(SPFA + DP)

    题目 Source http://www.lydsy.com/JudgeOnline/problem.php?id=2763 Description Alice和Bob现在要乘飞机旅行,他们选择了一家 ...

  7. Bzoj2763 [JLOI2011]飞行路线

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2651  Solved: 1004 Description Alice和Bob现在要乘飞机旅行,他们选 ...

  8. bzoj2763: [JLOI2011]飞行路线 分层图+dij+heap

    分析:d[i][j]代表从起点到点j,用了i次免费机会,那就可以最短路求解 #include <stdio.h> #include <iostream> #include &l ...

  9. [luogu4568][bzoj2763][JLOI2011]飞行路线

    题目描述 Alice和Bob现在要乘飞机旅行,他们选择了一家相对便宜的航空公司.该航空公司一共在n个城市设有业务,设这些城市分别标记为00到n-1,一共有m种航线,每种航线连接两个城市,并且航线有一定 ...

随机推荐

  1. Linux负载均衡概念与实践(一)

    根据网上文章整理. 负载均衡软件LVS(Linux Virtual Server)概念篇 lvs是在linux操作系统基础上建立虚拟服务器,实现服务节点之间的负载均衡.它是基于linux内核实现的.2 ...

  2. c#拖放

    AllowDrop DragEnter: if (e.Data.GetDataPresent(DataFormats.FileDrop)) e.Effect = DragDropEffects.Cop ...

  3. asp.net发送E-mail

    发送电子邮件也是项目开发当中经常用到的功能,这里我整理了一个发送电子邮件(带附件,支持多用户发送,主送.抄送)的类库,供大家参考. 先上两个实体类,用于封装成Mail对象. /// <summa ...

  4. jquery <li>标签 隔若干行 加空白或者加虚线

    $(function () { $('ul li').addClass(function (i) { return i % 6 == 5 ? "ab" : "" ...

  5. pc telnet 登录 android 系统

    前提是:1) 手机已经root,且装有busybox,2) 还装有至少一款terminal(模拟终端)软件,手机连wifi路由器.3) 还要有一些基础常识,比如linux命令,telnet.这里模拟终 ...

  6. mysql更改数据文件目录及my.ini位置| MySQL命令详解

    需求:更改mysql数据数据文件目录及my.ini位置. 步骤: 1.查找my.ini位置,可通过windows服务所对应mysql启动项,查看其对应属性->可执行文件路径,获取my.ini路径 ...

  7. Nginx+Keepalived主备负载均衡

    实验环境及软件版本: CentOS版本:    6.6(2.6.32.-504.el6.x86_64) nginx版本:     nginx-1.6.2 keepalived版本:keepalived ...

  8. 分享零基础学习Hadoop方法

    (我不是Hadoop专家,也只是一个初学者,这里我也只是就自己的学习体会,站在初学者的角度谈一下如何入门.) 首先我觉得应该思考这样一个问题:Hadoop对于我们来讲,是一种工具,那么Hadoop帮助 ...

  9. 一步步学习NHibernate(7)——HQL查询(1)

    请注明转载地址:http://www.cnblogs.com/arhat 从本章开始,老魏带着大家来学习一下HQL语句.HQL语句NHibernate为我们提供的一种功能比较强大的查询语句,这个HQL ...

  10. android打造万能的适配器(转)

    荒废了两天,今天与大家分享一个ListView的适配器 前段时间在学习慕课网的视频,觉得这种实现方式较好,便记录了下来,最近的项目中也使用了多次,节省了大量的代码,特此拿来与大家分享一下. 还是先看图 ...