正题

题目链接:https://loj.ac/p/2769


题目大意

给出\(n\)个点\(m\)条地铁线路,每条线路是一条路径。

求\(1\)到\(n\)的最短路且在最短路径的情况下相邻换乘点的距离平方和最大。

\(1\leq n,m,\sum s_i\leq 10^6\)


解题思路

首先肯定是在最短路树上跑,然后考虑怎么求平方和最大。

因为每条可以计算和的路径一定是连续的一段,所以考虑将一条铁路线去掉不在最短路树上的边后分出若干段来。

对于每一段上的转移都是一个\(f_{x}=max\{f_y+(d_x-d_y)^2\}\)的形式,可以斜率优化。

但是为了充分转移,需要按照\(dis\)从小到大转移。

栈啊什么的开\(vector\)就好了。

时间复杂度\(O(n\log n)\)


code

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. #include<vector>
  5. #include<queue>
  6. #include<cctype>
  7. #define mp(x,y) make_pair(x,y)
  8. #define sec second
  9. #define fir first
  10. #define ll long long
  11. using namespace std;
  12. const ll N=1e6+10;
  13. struct node{
  14. ll to,next,w;
  15. }a[N];
  16. ll n,m,T,tot,ls[N],dis[N],f[N],top[N],p[N],las[N];
  17. vector<pair<ll,ll> >e[N],E[N],pos[N];
  18. priority_queue<pair<ll,ll> >q;
  19. vector<ll> _x[N],_y[N],s[N];bool v[N];
  20. ll read(){
  21. ll x=0,f=1;char c=getchar();
  22. while(!isdigit(c)){if(c=='-')f=-f;c=getchar();}
  23. while(isdigit(c)){x=(x<<1)+(x<<3)+c-48;c=getchar();}
  24. return x*f;
  25. }
  26. void addl(ll x,ll y,ll w){
  27. a[++tot].to=y;
  28. a[tot].next=ls[x];
  29. ls[x]=tot;a[tot].w=w;
  30. return;
  31. }
  32. void dij(){
  33. memset(dis,0x3f,sizeof(dis));
  34. q.push(mp(0,1));dis[1]=0;
  35. while(!q.empty()){
  36. ll x=q.top().sec;q.pop();
  37. if(v[x])continue;v[x]=1;
  38. for(ll i=ls[x];i;i=a[i].next){
  39. ll y=a[i].to;
  40. if(dis[x]+a[i].w<dis[y]){
  41. dis[y]=dis[x]+a[i].w;
  42. q.push(mp(-dis[y],y));
  43. }
  44. }
  45. }
  46. return;
  47. }
  48. bool cmp(ll x,ll y)
  49. {return dis[x]<dis[y];}
  50. double slope(ll x,ll y,ll p)
  51. {return (double)(_y[p][y]-_y[p][x])/(_x[p][y]-_x[p][x]);}
  52. void calc(ll x,ll p,ll t,vector<ll> &s,ll &top){
  53. //x点 p线 t站
  54. if(!t)return;
  55. ll dx=e[p][t].sec;
  56. while(top>1&&slope(s[top-1],s[top],p)<=2*dx)top--;
  57. if(top){
  58. ll y=s[top];
  59. f[x]=max(f[x],_y[p][y]-2*dx*_x[p][y]+dx*dx);
  60. }
  61. }
  62. void add(ll x,ll p,ll t,vector<ll> &s,ll &top){
  63. //x点 p线 t站
  64. if(t>=e[p].size()-1)return;
  65. ll dx=e[p][t].sec;
  66. _y[p][t]=f[x]+dx*dx;_x[p][t]=dx;
  67. while(top>1&&slope(s[top-1],t,p)>=slope(s[top-1],s[top],p))top--;
  68. s[++top]=t;las[p]=t;return;
  69. }
  70. signed main()
  71. {
  72. // freopen("city.in","r",stdin);
  73. // freopen("city.out","w",stdout);
  74. n=read();m=read();
  75. for(ll i=1;i<=m;i++){
  76. ll num,x,t=0;num=read()+1;
  77. while(num--){
  78. if(E[i].size())t=read();x=read();
  79. if(E[i].size()){
  80. pair<ll,ll> z=E[i][E[i].size()-1];
  81. addl(z.fir,x,t);
  82. }//建边
  83. E[i].push_back(mp(x,t));
  84. }
  85. }
  86. dij();
  87. for(int i=1;i<=m;i++){
  88. for(int j=0;j<E[i].size();j++){
  89. int x=E[i][j].first,t=E[i][j].sec,flag=1;
  90. if((!j)||dis[E[i][j-1].fir]+E[i][j].sec!=dis[E[i][j].fir]){
  91. if(T&&e[T].size()==1){
  92. pos[e[T][0].first].pop_back();
  93. e[T][0].first=x;
  94. flag=0;
  95. }
  96. else T++,e[T].push_back(mp(x,0));
  97. }
  98. else e[T].push_back(mp(x,t));
  99. if(flag)s[T].push_back(0),_x[T].push_back(0),_y[T].push_back(0);
  100. pos[x].push_back(mp(T,e[T].size()-1));
  101. //e:线路(点/和前面的长度)
  102. //pos:点(线路,对应位置)
  103. //s,_x,_y:dp用数组
  104. }
  105. }
  106. for(ll i=1;i<=T;i++)
  107. for(ll j=1;j<e[i].size();j++)
  108. e[i][j].sec+=e[i][j-1].sec;
  109. for(ll i=1;i<=n;i++)p[i]=i;
  110. sort(p+1,p+1+n,cmp);
  111. for(ll _i=1;_i<=n;_i++){
  112. ll x=p[_i];
  113. for(ll _j=0;_j<pos[x].size();_j++)
  114. calc(x,pos[x][_j].fir,pos[x][_j].sec,s[pos[x][_j].fir],top[pos[x][_j].fir]);
  115. for(ll _j=0;_j<pos[x].size();_j++)
  116. add(x,pos[x][_j].fir,pos[x][_j].sec,s[pos[x][_j].fir],top[pos[x][_j].fir]);
  117. if(x==n)break;
  118. }
  119. printf("%lld %lld\n",dis[n],f[n]);
  120. return 0;
  121. }

Loj#2769-「ROI 2017 Day 1」前往大都会【最短路树,斜率优化】的更多相关文章

  1. LOJ #2769 -「ROI 2017 Day 1」前往大都会(单调栈维护斜率优化)

    LOJ 题面传送门 orz 斜率优化-- 模拟赛时被这题送走了,所以来写篇题解( 首先这个最短路的求法是 trivial 的,直接一遍 dijkstra 即可( 重点在于怎样求第二问.注意到这个第二问 ...

  2. [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞

    [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞 试题描述 到河北省 见斯大林 / 在月光下 你的背影 / 让我们一起跳舞吧 うそだよ~ 河北省怎么可能有 Stalin. ...

  3. loj #6250. 「CodePlus 2017 11 月赛」找爸爸

    #6250. 「CodePlus 2017 11 月赛」找爸爸 题目描述 小 A 最近一直在找自己的爸爸,用什么办法呢,就是 DNA 比对. 小 A 有一套自己的 DNA 序列比较方法,其最终目标是最 ...

  4. [LOJ 6249]「CodePlus 2017 11 月赛」汀博尔

    Description 有 n 棵树,初始时每棵树的高度为 H_i,第 i 棵树每月都会长高 A_i.现在有个木料长度总量为 S 的订单,客户要求每块木料的长度不能小于 L,而且木料必须是整棵树(即不 ...

  5. [LOJ 6248]「CodePlus 2017 11 月赛」晨跑

    Description “无体育,不清华”.“每天锻炼一小时,健康工作五十年,幸福生活一辈子” 在清华,体育运动绝对是同学们生活中不可或缺的一部分.为了响应学校的号召,模范好学生王队长决定坚持晨跑.不 ...

  6. loj 2392「JOISC 2017 Day 1」烟花棒

    loj 答案显然满足二分性,先二分一个速度\(v\) 然后显然所有没有点火的都会往中间点火的人方向走,并且如果两个人相遇不会马上点火,要等到火快熄灭的时候才点火,所以这两个人之后应该在一起行动.另外有 ...

  7. loj#2391 「JOISC 2017 Day 1」港口设施

    分析 https://yhx-12243.github.io/OI-transit/records/uoj356%3Bloj2391%3Bac2534.html 代码 #include<bits ...

  8. LOJ 2288「THUWC 2017」大葱的神力

    LOJ 2288「THUWC 2017」大葱的神力 Link Solution 比较水的提交答案题了吧 第一个点爆搜 第二个点爆搜+剪枝,我的剪枝就是先算出 \(mx[i]\) 表示选取第 \(i \ ...

  9. Loj #2731 「JOISC 2016 Day 1」棋盘游戏

    Loj 2731 「JOISC 2016 Day 1」棋盘游戏 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少 ...

随机推荐

  1. 熟悉而陌生的新朋友——IAsyncDisposable

    本文作者--句幽 在.NET Core 3.0的版本更新中,官方我们带来了一个新的接口 IAsyncDisposable. 小伙伴一看肯定就知道,它和.NET中原有的IDisposable接口肯定有着 ...

  2. Spring详解(四)------注解配置DI

    第一步:在 applicationContext.xml 中引入命名空间 这里我们简单讲解一下这里引入的命名空间,简单来说就是用来约束xml文件格式的.第一个 xmlns:context ,这表示标签 ...

  3. myScript调研,电子手写板使用,纯干货

    第二天进公司,就叫我调研myScript作为手写板的可行性,又不能不做,哎~ myScript效果十分的奈斯,前端用canvas手写的文字.数学字符,都可以识别然后转换,不知道myScript是不是你 ...

  4. Learning ROS: rqt_console和rqt_logger_level使用

    rqt_console:操作.查看log信息 rqt_logger_level:设置log等级 打开node: rosrun rqt_console rqt_console rosrun rqt_lo ...

  5. Python 脚本的执行

    源文件如下,文件名test.py,其中UTF-8根据实际情况而定,Python3默认为UTF-8,所以不用设置: #!/usr/bin/python # -*- coding: UTF-8 -*- p ...

  6. Gogs (Go git server) 使用笔记

    issue: 话题,一个新特性,BUG或其他关注的任何话题,都可创建issure,便于讨论,明确目标. label: 标签,一般用于描述issue的类型,如:bug.feature.enhanceme ...

  7. MySQL高可用主从复制简介

    原文转自:https://www.cnblogs.com/itzgr/p/10233932.html作者:木二 目录 一 简介 1.1 概述 二 技术原理 2.1 支持的复制类型 2.2 技术特点 2 ...

  8. 日常shell练习

    2021-07-19 1.echo的使用 1.1 echo -n 表示不换行输出 # echo输出会自动换行,换行输出两个1 echo 1 echo 1 # 不换行输出,不换行输出两个1 echo - ...

  9. QT学习日记篇01(1)-QT界面初探- *.pro文件详解

    一: 项目管理文件(.pro文件) 项目管理文件用于记录项目的一些设置,以及项目包含文件的组织管理 后缀为".pro"的 文件是项目的管理文件,文件名就是项目的名称,如Demo.p ...

  10. Django——Auth模块(用户认证模块)

    1.Auth模块简介 auth模块是对登录认证方法的一种封装,之前我们获取用户输入的用户名及密码后需要自己从user表里查询有没有用户名和密码符合的对象. 而有了auth模块之后就可以很轻松的去验证用 ...