\(\color{#0066ff}{ 题目描述 }\)

\(\color{#0066ff}{输入格式}\)

\(\color{#0066ff}{输出格式}\)

\(\color{#0066ff}{输入样例}\)

  1. 6 7
  2. 1 2 1
  3. 2 3 1
  4. 3 4 2
  5. 4 5 1
  6. 5 6 1
  7. 1 3 3
  8. 4 6 3
  9. 1 6
  10. 4
  11. 1 2
  12. 1 3
  13. 4 3
  14. 6 5

\(\color{#0066ff}{输出样例}\)

  1. 7
  2. 6
  3. Infinity
  4. 7

\(\color{#0066ff}{数据范围与提示}\)

\(\color{#0066ff}{ 题解 }\)

分别从s和t跑最短路,构建出最短路树

标记最短路树的点和边

从最短路树上的每个点bfs,找到能影响的L和R

显然若上图a,b之间的某条边断了,x到y的边就可以用来更新这部分答案

从a找到所有x,b找到所有y

枚举所有边,只要不在最短路树上,就类似于上图更新(用线段树维护)

在\(O(nlogn)\)的复杂度下求出删去每条最短路树上的边的ans

对于询问,如果不是最短路树的边,ans就是最短路

否则用刚刚在线段树求的ans输出

跑dij的pair要开long long!!!!

  1. #include<bits/stdc++.h>
  2. #define LL long long
  3. LL in() {
  4. char ch; LL x = 0, f = 1;
  5. while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
  6. for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
  7. return x * f;
  8. }
  9. const int maxn = 2e6 + 100;
  10. int L[maxn], R[maxn], S, T;
  11. int n, m;
  12. struct node {
  13. int to;
  14. LL dis;
  15. bool vis;
  16. node *nxt;
  17. node(int to = 0, LL dis = 0, bool vis = false, node *nxt = NULL): to(to), dis(dis), vis(vis), nxt(nxt) {}
  18. };
  19. node *head[maxn];
  20. bool vis[maxn];
  21. LL diss[maxn], dist[maxn];
  22. int st[maxn], cnt, rst[maxn];
  23. LL ans[maxn];
  24. const LL inf = 9999999999999LL;
  25. using std::pair;
  26. using std::make_pair;
  27. struct Tree {
  28. Tree *ch[2];
  29. LL val;
  30. int l, r;
  31. Tree(LL val = 0, int l = 0, int r = 0): val(val), l(l), r(r) {}
  32. }*root;
  33. void build(Tree *&o, int l, int r) {
  34. o = new Tree(inf, l, r);
  35. if(l == r) return;
  36. int mid = (l + r) >> 1;
  37. build(o->ch[0], l, mid);
  38. build(o->ch[1], mid + 1, r);
  39. }
  40. void add(int from, int to, LL dis) {
  41. head[from] = new node(to, dis, 0, head[from]);
  42. }
  43. void dij(int s, LL *dis) {
  44. std::priority_queue<pair<LL, int>, std::vector<pair<LL, int> >, std::greater<pair<LL, int> > > q;
  45. for(int i = 1; i <= n; i++) vis[i] = 0, dis[i] = inf;
  46. q.push(make_pair(dis[s] = 0, s));
  47. while(!q.empty()) {
  48. int tp = q.top().second;
  49. q.pop();
  50. if(vis[tp]) continue;
  51. vis[tp] = true;
  52. for(node *i = head[tp]; i; i = i->nxt)
  53. if(dis[i->to] > dis[tp] + i->dis)
  54. q.push(make_pair(dis[i->to] = dis[tp] + i->dis, i->to));
  55. }
  56. }
  57. void bfs(int s, int *P, LL *dis) {
  58. std::queue<int> v;
  59. P[st[s]] = s;
  60. v.push(st[s]);
  61. while(!v.empty()) {
  62. int tp = v.front(); v.pop();
  63. for(node *i = head[tp]; i; i = i->nxt) {
  64. if(dis[i->to] == dis[tp] + i->dis && !vis[i->to] && !P[i->to]) {
  65. P[i->to] = s;
  66. v.push(i->to);
  67. }
  68. }
  69. }
  70. }
  71. void change(Tree *o, int l, int r, LL val) {
  72. if(o->r < l || o->l > r) return;
  73. if(l <= o->l && o->r <= r) return (void)(o->val = std::min(o->val, val));
  74. change(o->ch[0], l, r, val), change(o->ch[1], l, r, val);
  75. }
  76. void query(Tree *o) {
  77. if(o->l == o->r) return (void)(ans[o->l] = o->val);
  78. o->ch[0]->val = std::min(o->ch[0]->val, o->val);
  79. o->ch[1]->val = std::min(o->ch[1]->val, o->val);
  80. query(o->ch[0]);
  81. query(o->ch[1]);
  82. }
  83. int main() {
  84. n = in(), m = in();
  85. LL x, y, z;
  86. for(int i = 1; i <= m; i++) {
  87. x = in(), y = in(), z = in();
  88. add(x, y, z), add(y, x, z);
  89. }
  90. dij(S = in(), diss);
  91. dij(T = in(), dist);
  92. for(int i = 1; i <= n; i++) vis[i] = 0;
  93. for(int o = S; o != T;) {
  94. st[rst[o] = ++cnt] = o;
  95. vis[o] = true;
  96. for(node *i = head[o]; i; i = i->nxt) {
  97. if(diss[o] + dist[i->to] + i->dis == diss[T]) {
  98. i->vis = true;
  99. o = i->to;
  100. break;
  101. }
  102. }
  103. }
  104. st[rst[T] = ++cnt] = T;
  105. vis[T] = true;
  106. for(int i = 1; i <= cnt; i++) bfs(i, L, diss);
  107. for(int i = cnt; i >= 1; i--) bfs(i, R, dist);
  108. build(root, 1, cnt);
  109. for(int i = 1; i <= n; i++)
  110. for(node *j = head[i]; j; j = j->nxt) {
  111. if(j->vis) continue;
  112. if(L[i] < R[j->to] && L[i] && R[j->to]) change(root, L[i], R[j->to] - 1, diss[i] + dist[j->to] + j->dis);
  113. }
  114. query(root);
  115. for(int q = in(); q --> 0;) {
  116. x = in(), y = in();
  117. if(rst[x] > 0 && rst[y] > 0 && abs(rst[x] - rst[y]) == 1) {
  118. LL t = ans[std::min(rst[x], rst[y])];
  119. if(t == inf) printf("Infinity\n");
  120. else printf("%lld\n", t);
  121. }
  122. else if(diss[T] == inf) printf("Infinty\n");
  123. else printf("%lld\n", diss[T]);
  124. }
  125. return 0;
  126. }

BZOJ 2725 [Violet 6]故乡的梦 线段树+最短路树的更多相关文章

  1. BZOJ 2725: [Violet 6]故乡的梦 最短路+线段树

    2725: [Violet 6]故乡的梦 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 678  Solved: 204[Submit][Status ...

  2. BZOJ 2725: [Violet 6]故乡的梦

    求出最短路径树,对于一个询问(x,y) 若不在树上S->T的链上,则答案不变,若在链上,考虑用一条非树边替换这条边,这条非树边必须跨越x->y这条边,线段树维护区间最小值 #include ...

  3. [原博客] BZOJ 2725 : [Violet 6]故乡的梦

    这个题在bzoj上好像是个权限题,想做的可以去Vani的博客下载测试数据.这里有题面. 简单叙述一下题意:给你一个n个点.m条边的带权无向图,S点和T点,询问Q次删一条给定的边的S-T最短路. 其中  ...

  4. [BZOJ 3123] [SDOI 2013]森林(可持久化线段树+并查集+启发式合并)

    [BZOJ 3123] [SDOI 2013]森林(可持久化线段树+启发式合并) 题面 给出一个n个节点m条边的森林,每个节点都有一个权值.有两种操作: Q x y k查询点x到点y路径上所有的权值中 ...

  5. BZOJ2725 : [Violet 6]故乡的梦

    如果S==T,那么答案为0. 如果S与T不连通,那么答案为inf. 否则,S到T的最短路径上至少有一条边. 求出以S为源点的最短路图,是个DAG,随便抓一条S到T的最短路,记为P. 设dpS[x]表示 ...

  6. BZOJ.5110.[CodePlus2017]Yazid 的新生舞会(线段树/树状数组/分治)

    LOJ BZOJ 洛谷 又来发良心题解啦 \(Description\) 给定一个序列\(A_i\).求有多少个子区间,满足该区间众数出现次数大于区间长度的一半. \(n\leq5\times10^5 ...

  7. BZOJ.2653.[国家集训队]middle(可持久化线段树 二分)

    BZOJ 洛谷 求中位数除了\(sort\)还有什么方法?二分一个数\(x\),把\(<x\)的数全设成\(-1\),\(\geq x\)的数设成\(1\),判断序列和是否非负. 对于询问\(( ...

  8. BZOJ 5168 && Luogu P3740 [HAOI2014]贴海报 线段树~~

    据说某谷数据十分水...但幸好BZOJ上也过了...话说我记得讲课时讲的是奇奇怪怪的离散化..但现在突然觉得什么都可以线段树瞎搞了...QAQ 直接就是这个区间有没有被覆盖,被覆盖直接return: ...

  9. BZOJ 3878 [AHOI&JSOI2014]奇怪的计算器 (线段树)

    题面:BZOJ传送门 洛谷传送门 线段树好题 题目保证$a$一定是正整数,容易发现计算结果是单调的 我们把询问离线,并按照从小到大排序 某次操作可能导致某些位置达到边界$L/R$ 根据单调性的结论 这 ...

随机推荐

  1. Oracle 数据库加密

    数据加密 动态数据(data in motion)和静态数据(data at rest),除了手动加密,其他的加密都需要oracle企业版的高级加密(额外收费——)  1 静态数据加密 Example ...

  2. mycat 分片

    1  配置下面两种ER分片,并结合日志分析子表插入过程中的不同 (1).父表按照主键ID分片,子表的分片字段与主表ID关联,配置为ER分片 (2).父表的分片字段为其他字段,子表的分片字段与主表ID关 ...

  3. PowerDesignerPDM中搜寻表名或字段名

    Option   Explicit ValidationMode   =   True InteractiveMode =   im_Batch Dim   mdl   '当前model '获取当前活 ...

  4. OpenGL渲染流水线

    其实OpenGL的流水线,对我学习来说只能算是一个概念性的东西.毕竟OpenGL也在发展,流水线也不会是一成不变的. 不过理解流水线的过程,重点在于理解每一步的作用,进而可以如何衔接起来,完成整个绘制 ...

  5. tomcat跑多个项目和不同端口访问项目

    最近笔者在工作中需要同时运行多个项目,且有时需要不同端口访问项目:在此过程中,笔者觉得有必要将注意事项记录一下,以备后边查阅或广大读者借鉴. 工作环境是win7,64位,IDE为eclipse,浏览器 ...

  6. C#连接MSSQL

    本文将介绍如何用C#连接MSSQL,C#连接SQL十分简单.我们一步一步来操作. 1.打开Microsoft SQL Server Management Studio创建一个数据库,这里我创建一个数据 ...

  7. JSP 按钮点击,onclick事件中的AJAX不执行可能的原因

    <button onclick="deleteAccount()"  >删除</button> 缺少了 type="button" &l ...

  8. sizeof总结

    1.sizeof常用总结 ①与strlen比较       strlen 计算字符串的字符数,以"\0"为结束判断,但不统计结束符.   sizeof 计算数据(数组.变量.类型. ...

  9. JVM实用参数(二)参数分类和即时(JIT)编译器诊断

    JVM实用参数(二)参数分类和即时(JIT)编译器诊断 作者: PATRICK PESCHLOW     原文地址    译者:赵峰 校对:许巧辉 在这个系列的第二部分,我来介绍一下HotSpot J ...

  10. cocos2d中setBlendFunc设置颜色混合方案

    CCSprite有一个ccBlendFunc类型的blendFunc_结构体成员,可以用来设置描绘时的颜色混合方案.ccBlendFunc包含了一个src和一个dst,分别表示源和目标的运算因子. 如 ...