Description

为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士。魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M。初始时小E同学在号节点1,隐士则住在号节点N。小E需要通过这一片魔法森林,才能够拜访到隐士。

魔法森林中居住了一些妖怪。每当有人经过一条边的时候,这条边上的妖怪就会对其发起攻击。幸运的是,在号节点住着两种守护精灵:A型守护精灵与B型守护精灵。小E可以借助它们的力量,达到自己的目的。

只要小E带上足够多的守护精灵,妖怪们就不会发起攻击了。具体来说,无向图中的每一条边Ei包含两个权值Ai与Bi。若身上携带的A型守护精灵个数不少于Ai,且B型守护精灵个数不少于Bi,这条边上的妖怪就不会对通过这条边的人发起攻击。当且仅当通过这片魔法森林的过程中没有任意一条边的妖怪向小E发起攻击,他才能成功找到隐士。

由于携带守护精灵是一件非常麻烦的事,小E想要知道,要能够成功拜访到隐士,最少需要携带守护精灵的总个数。守护精灵的总个数为A型守护精灵的个数与B型守护精灵的个数之和。

Input

第1行包含两个整数N,M,表示无向图共有N个节点,M条边。 接下来M行,第行包含4个正整数Xi,Yi,Ai,Bi,描述第i条无向边。其中Xi与Yi为该边两个端点的标号,Ai与Bi的含义如题所述。 注意数据中可能包含重边与自环。

Output

输出一行一个整数:如果小E可以成功拜访到隐士,输出小E最少需要携带的守护精灵的总个数;如果无论如何小E都无法拜访到隐士,输出“-1”(不含引号)。

Sample Input

【输入样例1】
4 5
1 2 19 1
2 3 8 12
2 4 12 15
1 3 17 8
3 4 1 17
【输入样例2】
3 1
1 2 1 1

Sample Output

【输出样例1】
32
【样例说明1】
如果小E走路径1→2→4,需要携带19+15=34个守护精灵;
如果小E走路径1→3→4,需要携带17+17=34个守护精灵;
如果小E走路径1→2→3→4,需要携带19+17=36个守护精灵;
如果小E走路径1→3→2→4,需要携带17+15=32个守护精灵。
综上所述,小E最少需要携带32个守护精灵。
【输出样例2】
-1
【样例说明2】
小E无法从1号节点到达3号节点,故输出-1。

HINT

2<=n<=50,000

0<=m<=100,000
1<=ai ,bi<=50,000

题解

$LCT$ 维护边权信息的题...但据说动态 $SPFA$ 能水过,我这么菜,像这种 $NOI$ 的题也只能靠水才能做得了...

贪心是把边按 $a$ 排序,动态加边,边权为 $b$ 。每加一条边,从边的两端点开始跑 $SPFA$ , $SPFA$ 维护路径上的瓶颈。然后取最后加的边的 $a$ 值 + $SPFA$ 过程中维护的起点到终点的最小化瓶颈值 $b$ 的和的最小值。

注意的是动态 $SPFA$ 和 [HNOI 2014]道路堵塞 是一样的,不要去清空 $dist$ 数组。

  1. //It is made by Awson on 2018.1.10
  2. #include <set>
  3. #include <map>
  4. #include <cmath>
  5. #include <ctime>
  6. #include <queue>
  7. #include <stack>
  8. #include <cstdio>
  9. #include <string>
  10. #include <vector>
  11. #include <cstdlib>
  12. #include <cstring>
  13. #include <iostream>
  14. #include <algorithm>
  15. #define LL long long
  16. #define lowbit(x) ((x)&(-(x)))
  17. #define Max(a, b) ((a) > (b) ? (a) : (b))
  18. #define Min(a, b) ((a) < (b) ? (a) : (b))
  19. #define Swap(a, b) ((a) ^= (b), (b) ^= (a), (a) ^= (b))
  20. using namespace std;
  21. const int N = ;
  22. const int M = ;
  23. const int INF = ~0u>>;
  24. void read(int &x) {
  25. char ch; bool flag = ;
  26. for (ch = getchar(); !isdigit(ch) && ((flag |= (ch == '-')) || ); ch = getchar());
  27. for (x = ; isdigit(ch); x = (x<<)+(x<<)+ch-, ch = getchar());
  28. x *= -*flag;
  29. }
  30. void write(int x) {
  31. if (x > ) write(x/);
  32. putchar(x%+);
  33. }
  34.  
  35. int n, m, ans = INF;
  36. struct ss {
  37. int u, v, a, b;
  38. bool operator < (const ss &tmp) const {
  39. return a < tmp.a;
  40. }
  41. }e[M+];
  42. struct tt {int to, next, cost; }edge[(M<<)+];
  43. int path[N+], top;
  44. int q[N+], head, tail, vis[N+], dist[N+];
  45. void add(int u, int v, int c) {edge[++top].to = v, edge[top].next = path[u], edge[top].cost = c, path[u] = top; }
  46. int SPFA(int u, int v) {
  47. vis[q[head = tail = ] = u] = , ++tail, vis[q[tail] = v] = , ++tail;
  48. while (head != tail) {
  49. int u = q[head]; ++head, head %= N, vis[u] = ;
  50. for (int i = path[u]; i; i = edge[i].next) {
  51. int v = edge[i].to;
  52. if (dist[v] > Max(dist[u], edge[i].cost)) {
  53. dist[v] = Max(dist[u], edge[i].cost);
  54. if (!vis[v]) vis[q[tail] = v] = , ++tail, tail %= N;
  55. }
  56. }
  57. }
  58. return dist[n];
  59. }
  60. void work() {
  61. read(n), read(m);
  62. for (int i = ; i <= m; i++) read(e[i].u), read(e[i].v), read(e[i].a), read(e[i].b);
  63. sort(e+, e++m); for (int i = ; i <= n; i++) dist[i] = INF;
  64. for (int i = ; i <= m; i++) {
  65. add(e[i].u, e[i].v, e[i].b), add(e[i].v, e[i].u, e[i].b);
  66. int tmp = SPFA(e[i].u, e[i].v); if (tmp != INF) ans = Min(ans, tmp+e[i].a);
  67. }
  68. if (ans != INF) write(ans);
  69. else putchar('-'), putchar('');
  70. }
  71. int main() {
  72. work();
  73. return ;
  74. }

动态SPFA

我已经这么菜了,肯定不能再怠惰...留个坑,用 $lct$ 再写一遍...

upd 18.1.11:填坑,码完 $lct$ 了。

对于边权问题的处理,用 $lct$ 不好简易地将边权转移为点权。解决这类问题的方法就是再开一倍点,将边变为点,表示边的点权就是原边权,而表示点的点权赋为 $0$ 。

对于这道题,方法和上面的一样,按照 $a$ 排序, $lct$ 维护最小瓶颈路。

  1. //It is made by Awson on 2018.1.11
  2. #include <set>
  3. #include <map>
  4. #include <cmath>
  5. #include <ctime>
  6. #include <queue>
  7. #include <stack>
  8. #include <cstdio>
  9. #include <string>
  10. #include <vector>
  11. #include <cstdlib>
  12. #include <cstring>
  13. #include <iostream>
  14. #include <algorithm>
  15. #define LL long long
  16. #define Max(a, b) ((a) > (b) ? (a) : (b))
  17. #define Min(a, b) ((a) < (b) ? (a) : (b))
  18. #define Swap(a, b) ((a) ^= (b), (b) ^= (a), (a) ^= (b))
  19. using namespace std;
  20. const int N = ;
  21. const int INF = ~0u>>;
  22.  
  23. int n, m, w[N+], a[N+], b[N+], ans = INF;
  24. struct tt {
  25. int u, v, a, b;
  26. bool operator < (const tt &tmp) const {
  27. return a < tmp.a;
  28. }
  29. }e[N+];
  30. struct Link_Cut_Tree {
  31. Link_Cut_Tree() {for (int i = ; i <= N; i++) isrt[i] = ; }
  32. int pre[N+], ch[N+][], rev[N+], maxi[N+], isrt[N+], pos;
  33. void pushup(int o) {
  34. if (!o) return;
  35. maxi[o] = o;
  36. if (w[maxi[ch[o][]]] > w[maxi[o]]) maxi[o] = maxi[ch[o][]];
  37. if (w[maxi[ch[o][]]] > w[maxi[o]]) maxi[o] = maxi[ch[o][]];
  38. }
  39. void pushdown(int o) {
  40. if (!rev[o] || !o) return;
  41. int ls = ch[o][], rs = ch[o][];
  42. Swap(ch[ls][], ch[ls][]), Swap(ch[rs][], ch[rs][]);
  43. rev[ls] ^= , rev[rs] ^= , rev[o] = ;
  44. }
  45. void push(int o) {if (!isrt[o]) push(pre[o]); pushdown(o); }
  46. void rotate(int o, int kind) {
  47. int p = pre[o];
  48. ch[p][!kind] = ch[o][kind], pre[ch[o][kind]] = p;
  49. if (isrt[p]) isrt[o] = , isrt[p] = ;
  50. else ch[pre[p]][ch[pre[p]][] == p] = o;
  51. pre[o] = pre[p];
  52. ch[o][kind] = p, pre[p] = o;
  53. pushup(p), pushup(o);
  54. }
  55. void splay(int o) {
  56. push(o);
  57. while (!isrt[o]) {
  58. if (isrt[pre[o]]) rotate(o, ch[pre[o]][] == o);
  59. else {
  60. int p = pre[o], kind = ch[pre[p]][] == p;
  61. if (ch[p][kind] == o) rotate(o, !kind), rotate(o, kind);
  62. else rotate(p, kind), rotate(o, kind);
  63. }
  64. }
  65. }
  66. void access(int o) {
  67. int y = ;
  68. while (o) {
  69. splay(o);
  70. isrt[ch[o][]] = , isrt[ch[o][] = y] = ;
  71. pushup(o); o = pre[y = o];
  72. }
  73. }
  74. void makeroot(int o) {access(o), splay(o); rev[o] ^= , Swap(ch[o][], ch[o][]); }
  75. int find(int o) {access(o), splay(o); while (ch[o][]) o = ch[o][]; return o; }
  76. void link(int x, int y) {makeroot(x); pre[x] = y; }
  77. void cut(int x, int y) {makeroot(x); access(y), splay(y); ch[y][] = pre[x] = , isrt[x] = ; pushup(y); }
  78. int query(int x, int y) {
  79. if (find(x)^find(y)) return INF;
  80. makeroot(x), access(y), splay(y);
  81. return maxi[y];
  82. }
  83. void update(int x, int y, int c) {
  84. int last = query(x, y);
  85. if (last == INF) {w[++pos] = c, a[pos] = x, b[pos] = y; link(x, pos), link(y, pos); return; }
  86. if (w[last] <= c) return;
  87. cut(last, a[last]), cut(last, b[last]);
  88. w[last] = c, a[last] = x, b[last] = y; link(x, last), link(y, last);
  89. }
  90. }T;
  91. void work() {
  92. scanf("%d%d", &n, &m); T.pos = n;
  93. for (int i = ; i <= m; i++) scanf("%d%d%d%d", &e[i].u, &e[i].v, &e[i].a, &e[i].b);
  94. sort(e+, e++m);
  95. for (int i = ; i <= m; i++) {
  96. T.update(e[i].u, e[i].v, e[i].b);
  97. int tmp = T.query(, n); if (tmp != INF) ans = Min(ans, e[i].a+w[tmp]);
  98. }
  99. if (ans == INF) ans = -; printf("%d\n", ans);
  100. }
  101. int main() {
  102. work();
  103. return ;
  104. }

link-cut-tree

[NOI 2014]魔法森林的更多相关文章

  1. 【BZOJ3669】[Noi2014]魔法森林 LCT

    终于不是裸的LCT了...然而一开始一眼看上去这是kruskal..不对,题目要求1->n的路径上的每个点的两个最大权值和最小,这样便可以用LCT来维护一个最小生成路(瞎编的...),先以a为关 ...

  2. BZOJ 3669 【NOI2014】 魔法森林

    Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节 ...

  3. BZOJ-3669 魔法森林 Link-Cut-Tree

    意识到背模版的重要性了,记住了原理和操作,然后手打模版残了..颓我时间...... 3669: [Noi2014]魔法森林 Time Limit: 30 Sec Memory Limit: 512 M ...

  4. 【BZOJ】3669: [Noi2014]魔法森林(lct+特殊的技巧)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3669 首先看到题目应该可以得到我们要最小化 min{ max{a(u, v)} + max{b(u, ...

  5. [NOI 2014]做题记录

    [NOI2014]起床困难综合症 按位贪心 #include <algorithm> #include <iostream> #include <cstring> ...

  6. NOI2014 魔法森林

    3669: [Noi2014]魔法森林 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 106  Solved: 62[Submit][Status] ...

  7. bzoj 3669: [Noi2014]魔法森林 动态树

    3669: [Noi2014]魔法森林 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 363  Solved: 202[Submit][Status] ...

  8. 图论 BZOJ 3669 [Noi2014]魔法森林

    Description 为了得到书法大家的真传,小E同学下定决心去拜访住在魔法森林中的隐士.魔法森林可以被看成一个包含个N节点M条边的无向图,节点标号为1..N,边标号为1..M.初始时小E同学在号节 ...

  9. BZOJ 3669: [Noi2014]魔法森林( LCT )

    排序搞掉一维, 然后就用LCT维护加边MST. O(NlogN) ------------------------------------------------------------------- ...

随机推荐

  1. python全栈学习--day11(函数高级应用)

    一,函数名是什么? 函数名是函数的名字,本质:变量,特殊的变量. 函数名()执行此函数 ''' 在函数的执行(调用)时:打散. *可迭代对象(str,tuple,list,dict(key))每一个元 ...

  2. 福州大学W班-团队作业-随堂小测(同学录)成绩

    作业链接 https://edu.cnblogs.com/campus/fzu/FZUSoftwareEngineering1715W/homework/1246 作业要求 1.题目 即编写一个能够记 ...

  3. 第一次作业:扑通扑通 我的IT

    让我掉下眼泪的不止昨夜的酒,还有这满屏的代码. 第一部分:结缘计算机 你为什么选择计算机专业?你认为你的条件如何?和这些博主比呢? 在炎炎的夏日,伴随这高三的结束,我也面临大学专业的选择,我看着书里密 ...

  4. Exception in thread "main" expected '<document start>', but found BlockMappingStart in 'reader', line 23, column 2: nimbus.host: "master"

    平台:centos-6.3-i386 jdk-7u51 storm 0.9.1 python 2.6.6   hadoop 1.2.1 启动storm的时候,遇到这个问题,百度之后,看到大家的解决方案 ...

  5. JAVA_SE基础——17.方法的重载

    方法重载: 方法重载就是方法名称重复,加载参数不同. 具体规范: 一.方法名一定要相同. 二.方法的参数表必须不同,包括参数的类型或个数,以此区分不同的方法体. 1.如果参数个数不同,就不管它的参数类 ...

  6. php的控制器链

    控制器之间协同工作就形成了控制器链· 比如在一个控制器的方法中,创建另外一个·控制器,创建对象,然后调用第二个控制器方法,那么在第一个控制器分配给视图的变量,在 第二个控制器的方法中对应的视图也是可以 ...

  7. php的set_time_limit()函数

    set_time_limit(0); 括号里边的数字是执行时间,如果为零说明永久执行直到程序结束,如果为大于零的数字,则不管程序是否执行完成,到了设定的秒数,程序结束. 一个简单的例子,在网页里显示1 ...

  8. Docker加速器(阿里云)

    1. 登录阿里开发者平台: https://dev.aliyun.com/search.html,https://cr.console.aliyun.com/#/accelerator,生成专属链接 ...

  9. angular2 学习笔记 ( Rxjs, Promise, Async/Await 的区别 )

    Promise 是 ES 6 Async/Await 是 ES 7 Rxjs 是一个 js 库 在使用 angular 时,你会经常看见这 3 个东西. 它们都和异步编程有关,有些情况下你会觉得用它们 ...

  10. oracle11g导出表时会发现少表,空表导不出解决方案。

    一:背景引入 oracle11g用exp命令导出数据库表时,有时会发现只导出了一部分表时而且不会报错,原因是有空表没有进行导出,之前一直没有找到方法于是用最笨的方法重新建这些空表,当然在我们实际当中表 ...