题目背景

SHOI2012 D2T1

题目描述

2046 年 OI 城的城市轨道交通建设终于全部竣工,由于前期规划周密,建成后的轨道交通网络由2n2n条地铁线路构成,组成了一个nn纵nn横的交通网。如下图所示,这2n2n条线路每条线路都包含nn个车站,而每个车站都在一组纵横线路的交汇处。

出于建设成本的考虑,并非每个车站都能够进行站内换乘,能够进行站内换乘的地铁站共有mm个,在下图中,标上方块标记的车站为换乘车站。已知地铁运行 1 站需要 2 分钟,而站内换乘需要步行 1 分钟。Serenade 想要知道,在不中途出站的前提下,他从学校回家最快需要多少时间(等车时间忽略不计)。

输入输出格式

输入格式:

第一行有两个整数n,mn,m。

接下去mm行每行两个整数x,yx,y,表示第xx条横向线路与第yy条纵向线路的交

汇站是站内换乘站。

接下去一行是四个整数x_1,y_1,x_2,y_2x1,y1,x2,y2。表示 Serenade 从学校回家时,在第 x_1x1条横向线路与第y_1y1条纵向线路的交汇站上车,在第x_2x2条横向线路与第y_2y2条纵向线路的交汇站下车。

输出格式:

输出文件只有一行,即 Serenade 在合理选择线路的情况下,回家所需要的时间。如果 Serenade 无法在不出站换乘的情况下回家,请输出-1。

说明

对于 30%的数据,n\le 50,m\le 1000n≤50,m≤1000;

对于 60%的数据,n\le 500,m\le 2000n≤500,m≤2000;

对于 100%的数据,n\le 20000,m\le 100000n≤20000,m≤100000;


对于每行每列暴力连边显然是要炸的

考虑只把相邻的点间连边,对于同行同列是没有影响的

然鹅转向就跪了

经过一番深思熟虑,我们可以把每个点拆开啊

对于每个点,横向建一个点,纵向建一个点,把这两个点间连边\(edge=1\)

然后最短路就行了


  1. #include<iostream>
  2. #include<cstdio>
  3. #include<algorithm>
  4. #include<cstring>
  5. #include<queue>
  6. #define M 1000000
  7. #define LL long long
  8. #define max(a,b) ((a)>(b)? (a):(b))
  9. #define min(a,b) ((a)<(b)? (a):(b))
  10. #define abs(a) ((a)>0? (a) :-(a))
  11. using namespace std;
  12. queue<int> q;
  13. int i,m,n,j,k,ver[M],edge[M],head[M],nex[M],cnt,pre[M],sx,sy,ex,ey,b[M],d[M];
  14. struct vv
  15. {
  16. int x,y,w;
  17. } a[M];
  18. void add(int x,int y,int z)
  19. {
  20. cnt+=1;
  21. ver[cnt]=y, nex[cnt]=head[x], head[x]=cnt, edge[cnt]=z;
  22. }
  23. bool cmpx(vv a,vv b){return a.x==b.x? a.y<b.y : a.x<b.x;}
  24. bool cmpy(vv a,vv b){return a.y==b.y? a.x<b.x : a.y<b.y;}
  25. void spfa()
  26. {
  27. memset(d,0x3f,sizeof(d));
  28. d[0]=b[0]=1;
  29. q.push(0);
  30. while(q.size())
  31. {
  32. int r=q.front(); q.pop();
  33. b[r]=0;
  34. for(int i=head[r];i;i=nex[i])
  35. {
  36. int t=ver[i];
  37. if(d[r]+edge[i]<d[t])
  38. {
  39. d[t]=d[r]+edge[i];
  40. if(!b[t]) q.push(t);
  41. b[t]=1;
  42. }
  43. }
  44. }
  45. }
  46. int main()
  47. {
  48. scanf("%d%d",&n,&m);
  49. for(i=1;i<=m;i++) scanf("%d%d",&a[i].x,&a[i].y), a[i].w=i;
  50. scanf("%d%d%d%d",&sx,&sy,&ex,&ey);
  51. sort(a+1,a+1+m,cmpx);
  52. for(i=2;i<=m;i++)
  53. {
  54. if(a[i].x==a[i-1].x) add(i,i-1,(a[i].y-a[i-1].y)*2), add(i-1,i,(a[i].y-a[i-1].y)*2);
  55. if(a[i].x==sx) add(0,i,abs(a[i].y-sy)*2);
  56. if(a[i].x==ex) add(i,m+1+m,abs(a[i].y-ey)*2);
  57. }
  58. sort(a+1,a+1+m,cmpy);
  59. for(i=2;i<=m;i++)
  60. {
  61. if(a[i].x==a[i-1].x) add(i+m,i-1+m,(a[i].x-a[i-1].x)*2), add(i-1+m,i+m,(a[i].x-a[i-1].x)*2);
  62. if(a[i].y==sy) add(0,i+m,abs(a[i].x-sx)*2);
  63. if(a[i].y==ey) add(i+m,m+1+m,abs(a[i].x-ex)*2);
  64. }
  65. for(i=1;i<=m;i++) add(i,i+m,1), add(i+m,i,1);
  66. spfa();
  67. printf("%d",d[m+m+1]);
  68. }

[SHOI2012]回家的路的更多相关文章

  1. P3831 [SHOI2012]回家的路

    P3831 [SHOI2012]回家的路 分层图基础题,就是建图稍有麻烦   #include<cstdio> #include<algorithm> #include< ...

  2. [SHOI2012]回家的路 最短路

    ---题面--- 题解: 吐槽:找了好久的错,换了n种方法,重构一次代码,,,, 最后发现,,, 数组开小了,其实一开始尝试开大了数组,但唯独没有尝试开大手写队列的数组.... 思路: 有两种方法,这 ...

  3. 题解 P3831 [SHOI2012]回家的路

    什么叫分层图最短路,我不会/kk 感觉自己做法和其他题解不大一样所以过来发篇题解了. 未刻意卡常拿下最优解 题目大意 就是说给你一个 \(n \times n\) 的网格图和 \(m\) 个可换乘点, ...

  4. Bzoj 2834: 回家的路 dijkstra,堆优化,分层图,最短路

    2834: 回家的路 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 62  Solved: 38[Submit][Status][Discuss] D ...

  5. 分层图最短路【bzoj2834】: 回家的路

    分层图最短路[bzoj2834]: 回家的路 题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2834 这道题难在建边. 自己写的时候想到了 ...

  6. bzoj 2834: 回家的路

    题目 F.A.Qs Home Discuss ProblemSet Status Ranklist Contest 入门OJ ModifyUser  DCOI Logout 捐赠本站 Notice:1 ...

  7. 洛谷P3831 回家的路

    题目背景 SHOI2012 D2T1 题目描述 \(2046\) 年 \(OI\) 城的城市轨道交通建设终于全部竣工,由于前期规划周密,建成后的轨道交通网络由\(2n\)条地铁线路构成,组成了一个\( ...

  8. BZOJ.2834.回家的路(最短路Dijkstra 拆点)

    题目链接 对于相邻的.处在同在一行或一列的车站连边,然后用dis[x][0/1](或者拆点)分别表示之前是从横边还是竖边到x的,跑最短路. 我选择拆点.. //13028kb 604ms #inclu ...

  9. 解题:SHOI 2012 回家的路

    题面 完了,做的时候已经想不起来分层图这个东西了QAQ 对于这种“多种”路径加中转站的题,还有那种有若干次“特殊能力”的题,都可以考虑用分层图来做 显然只需要记录所有的中转站+起点终点,然后拆出横竖两 ...

随机推荐

  1. [javaSE] 数组(获取最值)

    数组的常见操作(获取最值) 1.获取最值需要进行比较,每一次比较都会有一个较大的值,因为该值不确定,通过一个变量进行存储 2.让数组中的每一个元素都和这个变量中的值进行比较,如果大于了变量中的值,就用 ...

  2. [javascript] Promise简单学习使用

    原文地址:http://www.cnblogs.com/dojo-lzz/p/4340897.html 解决回调函数嵌套太深,并行逻辑必须串行执行,一个Promise代表一个异步操作的最终结果,跟Pr ...

  3. solr+tomcat整合

    一.solr安装 http://archive.apache.org/dist/lucene/solr/ 这个地址有各个版本的 这次我使用的是5.5.4版本和tomcat8 版本5.5.4已经内置了j ...

  4. c#winform中如何修改ListView控件每个单元格的颜色

    ListView在View属性为Details的时候它呈现的方式和DataGirdView差不多,它的每一行都是一个ListViewItem,然而每个ListViewItem都有一个BackColor ...

  5. bootstrapValidator验证中Maximum call stack size exceeded

    Tip1:如果表单不是通过Bootstrap构建(即元素包含表单项且关联的label没有form-group类),可能会看到错误Uncaught RangeError: Maximum call st ...

  6. opencv图像处理时使用文件输入流ifstream批量读取图片

    简介: 在利用opencv进行图像处理时,通常需要批量读取图片,然后做相应的处理,我们可以用C++文件的输入流来进行图片的读取,这要求我们应该事先,将图片图片名生成txt文件,具体请参见之前的博文[u ...

  7. Myeclipse使用过程配置汇总

    1.下载安装及破解方法 myeclipse2014专业版下载地址链接:https://pan.baidu.com/s/1i62YOGt 密码:nlqj    : 下载后安装到最后一步先不要打开软件,如 ...

  8. RPC簡介

    RPC 技术原理       RPC ( Remote Procedure Call Protocol,远程过程调用协议 ): 客户端在不知道调用细节的情况下,调用存在于远程计算机上的某个对象,就像调 ...

  9. Week7——JSON

    1.什么是JSON? JSON是JavaScript对象表示法,全称是JavaScript Objec Notation.它是存储和交换文本信息的语法,类似于XML.JSON是轻量级的文本数据交换格式 ...

  10. Ionic Css样式详解

    Header是固定在屏幕顶部的组件.可以包含如标题和左右的功能按钮.Sub Header同样是固定在顶部,只是是在Header的下面,就算没有写Header这个,Sub Header这个样式也会距离顶 ...