题目链接: BZOJ - 1576

题目分析

首先Orz Hzwer的题解

先使用 dijikstra 求出最短路径树。

那么对于一条不在最短路径树上的边 (u -> v, w) 我们可以先沿树边从 1 走到 u ,再走这条边到 v ,然后再沿树边向上,可以走到 (LCA(u, v), v] 的所有点 (不包括LCA(u, v)!!)。

对于一个属于 (LCA(u, v), v] 的点 x,这种走法的距离为 d[u] + w + d[v] - d[x] ,那么我们就可以用 d[u] + w + d[v] 更新 (LCA(u, v), v] 这一段点的权值,使用树链剖分 + 线段树。

枚举每一条非树边进行更新。

最后每个点 x 的答案就是 x 的权值 - d[x] 。

注意!LCA(u, v) 是不能被这条边更新的!

代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <cstdlib>
#include <algorithm>
#include <queue> using namespace std; const int MaxN = 100000 + 5, MaxM = 200000 + 5, MaxLog = 20, INF = 999999999; int n, m, Index;
int Father[MaxN], Depth[MaxN], Top[MaxN], Size[MaxN], Son[MaxN], Pos[MaxN];
int d[MaxN], D[MaxN * 4], Jump[MaxN][MaxLog + 3]; struct Edge
{
int u, v, w;
bool Mark;
Edge *Next;
} E[MaxM * 2], *P = E, *Pre[MaxN], *Point[MaxN]; inline void AddEdge(int x, int y, int z) {
++P; P -> u = x; P -> v = y; P -> w = z; P -> Mark = false;
P -> Next = Point[x]; Point[x] = P;
} struct ES
{
int x, y;
ES() {}
ES(int a, int b) {
x = a; y = b;
}
}; struct Cmp
{
bool operator () (ES a, ES b) {
return a.y > b.y;
}
}; priority_queue<ES, vector<ES>, Cmp> Q; bool Visit[MaxN]; void Dijkstra() {
while (!Q.empty()) Q.pop();
for (int i = 1; i <= n; ++i) {
d[i] = INF; Visit[i] = false;
}
d[1] = 0;
for (int i = 1; i <= n; ++i) Q.push(ES(i, d[i]));
ES Now;
int x;
while (!Q.empty()) {
Now = Q.top(); Q.pop();
x = Now.x;
if (Visit[x]) continue;
Visit[x] = true;
for (Edge *j = Point[x]; j; j = j -> Next) {
if (d[x] + (j -> w) < d[j -> v]) {
d[j -> v] = d[x] + j -> w;
if (Pre[j -> v] != NULL) Pre[j -> v] -> Mark = false;
Pre[j -> v] = j;
j -> Mark = true;
Q.push(ES(j -> v, d[j -> v]));
}
}
}
} int DFS_1(int x, int Dep, int Fa) {
Depth[x] = Dep; Father[x] = Fa;
Size[x] = 1;
int SonSize, MaxSonSize;
SonSize = MaxSonSize = 0;
for (Edge *j = Point[x]; j; j = j -> Next) {
if (j -> v == Fa || j -> Mark == false) continue;
SonSize = DFS_1(j -> v, Dep + 1, x);
if (SonSize > MaxSonSize) {
MaxSonSize = SonSize;
Son[x] = j -> v;
}
Size[x] += SonSize;
}
return Size[x];
} void DFS_2(int x) {
if (x == 0) return;
if (x == Son[Father[x]]) Top[x] = Top[Father[x]];
else Top[x] = x;
Pos[x] = ++Index;
DFS_2(Son[x]);
for (Edge *j = Point[x]; j; j = j -> Next) {
if (j -> v == Father[x] || j -> v == Son[x] || j -> Mark == false) continue;
DFS_2(j -> v);
}
} void Build_Tree(int x, int s, int t) {
D[x] = INF;
if (s == t) return;
int m = (s + t) >> 1;
Build_Tree(x << 1, s, m);
Build_Tree(x << 1 | 1, m + 1, t);
} void Init_LCA() {
for (int i = 1; i <= n; ++i) Jump[i][0] = Father[i];
for (int j = 1; j <= MaxLog; ++j) {
for (int i = 1; i <= n; ++i) {
if (Depth[i] < (1 << j)) continue;
Jump[i][j] = Jump[Jump[i][j - 1]][j- 1];
}
}
} int LCA(int x, int y) {
int Dif;
if (Depth[x] < Depth[y]) swap(x, y);
Dif = Depth[x] - Depth[y];
if (Dif) {
for (int i = 0; i <= MaxLog; ++i) {
if (Dif & (1 << i)) x = Jump[x][i];
}
}
if (x == y) return x;
for (int i = MaxLog; i >= 0; --i) {
if (Jump[x][i] != Jump[y][i]) {
x = Jump[x][i];
y = Jump[y][i];
}
}
return Father[x];
} inline int gmin(int a, int b) {return a < b ? a : b;} void Paint(int x, int Num) {
if (Num >= D[x]) return;
D[x] = Num;
} void PushDown(int x) {
if (D[x] == INF) return;
Paint(x << 1, D[x]);
Paint(x << 1 | 1, D[x]);
D[x] = INF;
} void Change(int x, int s, int t, int l, int r, int Num) {
if (l <= s && r >= t) {
Paint(x, Num);
return;
}
PushDown(x);
int m = (s + t) >> 1;
if (l <= m) Change(x << 1, s, m, l, r, Num);
if (r >= m + 1) Change(x << 1 | 1, m + 1, t, l, r, Num);
} void EChange(int x, int y, int z) {
int fx, fy;
fx = Top[x]; fy = Top[y];
while (fx != fy) {
Change(1, 1, n, Pos[fx], Pos[x], z);
x = Father[fx];
fx = Top[x];
}
if (x != y) Change(1, 1, n, Pos[y] + 1, Pos[x], z);
} int Get(int x, int s, int t, int p) {
if (s == t) return D[x];
PushDown(x);
int m = (s + t) >> 1;
int ret;
if (p <= m) ret = Get(x << 1, s, m, p);
else ret = Get(x << 1 | 1, m + 1, t, p);
return ret;
} int main()
{
scanf("%d%d", &n, &m);
int a, b, c;
for (int i = 1; i <= m; ++i) {
scanf("%d%d%d", &a, &b, &c);
AddEdge(a, b, c);
AddEdge(b, a, c);
}
Dijkstra();
DFS_1(1, 0, 0);
Index = 0;
DFS_2(1);
Build_Tree(1, 1, n);
Init_LCA();
int t;
for (Edge *j = E + 1; ; ++j) {
if (j -> Mark) continue;
t = LCA(j -> u, j -> v);
EChange(j -> v, t, d[j -> u] + j -> w + d[j -> v]);
if (j == P) break;
}
int Temp;
for (int i = 2; i <= n; ++i) {
Temp = Get(1, 1, n, Pos[i]);
if (Temp < INF) printf("%d\n", Temp - d[i]);
else printf("-1\n");
}
return 0;
}

  

[BZOJ 1576] [Usaco2009 Jan] 安全路经Travel 【树链剖分】的更多相关文章

  1. bzoj 1576: [Usaco2009 Jan]安全路经Travel 树链剖分

    1576: [Usaco2009 Jan]安全路经Travel Time Limit: 10 Sec  Memory Limit: 64 MB Submit: 665  Solved: 227[Sub ...

  2. BZOJ1576: [Usaco2009 Jan]安全路经Travel(树链剖分)

    Description Input * 第一行: 两个空格分开的数, N和M * 第2..M+1行: 三个空格分开的数a_i, b_i,和t_i Output * 第1..N-1行: 第i行包含一个数 ...

  3. bzoj 1576 [Usaco2009 Jan]安全路经Travel(树链剖分,线段树)

    [题意] 给定一个无向图,找到1-i所有的次短路经,要求与最短路径的最后一条边不重叠. [思路] 首先用dijkstra算法构造以1为根的最短路树. 将一条无向边看作两条有向边,考察一条不在最短路树上 ...

  4. bzoj 1576: [Usaco2009 Jan]安全路经Travel【spfa+树链剖分+线段树】

    这几天写USACO水题脑子锈住了--上来就贪心,一交就WA 事实上这个是一个叫最短路树的东西,因为能保证只有一条最短路,所以所有最短路合起来是一棵以1为根的树,并且在这棵树上,每个点被精灵占据的路是它 ...

  5. BZOJ.1576.[Usaco2009 Jan]安全路经Travel(树形DP 并查集)

    题目链接 BZOJ 洛谷 先求最短路树.考虑每一条非树边(u,v,len),设w=LCA(u,v),这条边会对w->v上的点x(x!=w)有dis[u]+dis[v]-dis[x]+len的距离 ...

  6. bzoj 1576: [Usaco2009 Jan]安全路经Travel——并查集+dijkstra

    Description Input * 第一行: 两个空格分开的数, N和M * 第2..M+1行: 三个空格分开的数a_i, b_i,和t_i Output * 第1..N-1行: 第i行包含一个数 ...

  7. BZOJ 1576: [Usaco2009 Jan]安全路经Travel

    日常自闭半小时后看题解,太弱了qwq. 感觉这道题还是比较难的,解法十分巧妙,不容易想到. 首先题目说了起点到每个点的最短路都是唯一的,那么对这个图求最短路图必定是一棵树,而且这棵树是唯一的. 那么我 ...

  8. 【BZOJ】1576 [Usaco2009 Jan]安全路经Travel

    [算法]最短路树+(树链剖分+线段树)||最短路树+并查集 [题解] 两种方法的思想是一样的,首先题目限制了最短路树唯一. 那么建出最短路树后,就是询问对于每个点断掉父边后重新找路径的最小值,其它路径 ...

  9. 【BZOJ1576】[Usaco2009 Jan]安全路经Travel 最短路+并查集

    [BZOJ1576][Usaco2009 Jan]安全路经Travel Description Input * 第一行: 两个空格分开的数, N和M * 第2..M+1行: 三个空格分开的数a_i, ...

随机推荐

  1. WinForm中TextBox 中判断扫描枪输入与键盘输入

    本文转载:http://www.cnblogs.com/Hdsome/archive/2011/10/28/2227712.html 提出问题:在收货系统中,常常要用到扫描枪扫描条码输入到TextBo ...

  2. [ACM] HDU 5025 Saving Tang Monk (状态压缩,BFS)

    Saving Tang Monk Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  3. 程序猿接私活经验总结,来自csdn论坛语录

    下面为网上摘录,以做笔记: 但是到网上看看,似乎接私活也有非常多不easy,技术问题本身是个因素,还有非常多有技术的人接私活时被骗,或者是合作到最后以失败告终,所以想请有经验的大侠们出来指点一下,接私 ...

  4. docker registry 搭建

    registry 搭建 假设在 cd /root 目录下 最新 docker run -d -p 5000:5000 -v pwd/data:/var/lib/registry --restart=a ...

  5. export 命令 设置环境变量

    export KERN_DIR=/usr/src/kernels/2.6.18-194.11.1.el5-x86_64 设置环境变量 内核加载目录

  6. 【Spring五】AOP之使用注解配置

    AOP使用注解配置流程: 1.当spring容器启动时候.    < context:component- scan base-package= "cn.itheima03.sprin ...

  7. RPM命令用法详解

    RPM 有五种基本的操作方式(不包括创建软件包): 安装, 卸载, 升级, 查询,和验证. 下面我们就来逐一的讲解吧. 一. 安装RPM包 RPM 软件包通常具有类似foo-1.0-1.i386.rp ...

  8. 用 Qt 中的 QDomDocument类 处理 XML 文件(下)

      QDomDocument doc; 1).创建根节点:QDomElement root = doc.documentElement("rootName " ); 2).创建元素 ...

  9. ASP.NET MVC 第五回 ActionResult的其它返回值

    我们上边所看到的Action都是return View();我们可以看作这个返回值用于解析一个aspx文件.而它的返回类型是ActionResult如 public ActionResult Inde ...

  10. 使用Convert 类和Parse方法将字符串转换为数值类型

    //用Parse方法将字符串转换为数值类型; long num=Int64.Parse(args[2]) //用别名为Int64c#类型long; long num=long.Parse(args[2 ...