[HNOI2014] 道路堵塞 - 最短路,线段树
对不起对不起,辣鸡蒟蒻又来用核弹打蚊子了
完全ignore了题目给出的最短路,手工搞出一个最短路,发现对答案没什么影响
所以干脆转化为经典问题:每次询问删掉一条边后的最短路
如果删掉的是非最短路边,那么显然毫无影响
如果删掉的是最短路边,那么我们倒过来,考虑这个时候每条非最短路边的贡献。对于一条非最短路边 \((u,v)\) ,我们很容易得到一定包含它的最短路一定是满足 \(1 \to x \to u \to v \to y \to n\) 这样的结构,其中 \(x,y\) 都在选定的最短路上,那么只要删除的是最短路上 \(x \sim y\) 之间的边,这个贡献就是存在的。
考虑如何求出对 \((u,v)\) 的 \(x,y\) ,跑完最短路后搞出最短路径树,那么最短路显然是最短路径树的一条链。对这条链上任意一个点 \(p\) ,对它所有不在链上的儿子 \(q\) ,我们将每一个 \(q\) 的子树都打上 \(p\) 标记,表示从 \(1\) 到这些点的最短路在 \(p\) 点与选定的 \(1 \sim n\) 最短路分开
因此我们维护删掉最短路上每条边时的答案,这是一个序列。枚举每条非最短路边去计算它的贡献与贡献范围,贡献范围内的每个位置对它的贡献值取Min,线段树标记永久化维护即可。
#include <bits/stdc++.h>
using namespace std;
const int N = 400005;
int cnt = 0;
struct item {
int v,w,id;
};
struct graph {
vector <item> g[N];
vector <int> tree[N];
int d[N],v[N],c[N],fa[N],fid[N];
void make(int t1,int t2,int t3,int t4) {
g[t1].push_back((item) {
t2,t3,t4
});
}
void sp(int v0,int n) {
memset(d,0x3f,sizeof d);
memset(v,0x00,sizeof v);
priority_queue <pair<int,int> > q;
d[v0]=0;
q.push(make_pair(0,v0));
while(!q.empty()) {
pair<int,int> p=q.top();
int dis=-p.first, pos=p.second;
q.pop();
v[pos]=1;
for(int i=0;i<g[pos].size();i++) {
int x=g[pos][i].v,y=g[pos][i].w,z=g[pos][i].id;
if(d[x]>d[pos]+y) {
d[x]=d[pos]+y;
fa[x]=pos;
fid[x]=z;
if(!v[x]) q.push(make_pair(-d[x],x));
}
}
}
for(int i=1; i<=n; i++) {
if(fa[i])
tree[fa[i]].push_back(i);
c[i]=i;
}
}
void dfs(int p,int x) {
c[p]=x;
for(int i=0; i<tree[p].size(); i++) {
dfs(tree[p][i],x);
}
}
void gen(int s,int t) {
for(int p=t,q; q=fa[p]; p=q) {
for(int i=0; i<tree[q].size(); i++) {
if(tree[q][i]!=p) {
dfs(tree[q][i],q);
}
}
}
}
} gs,gt;
struct edge {
int u,v,w;
};
edge e[N];
int n,m,q,t1,t2,t3,t4,use[N],sid[N];
vector <int> sp;
namespace seg {
int a[N<<2];
void reset() {
memset(a,0x3f,sizeof a);
}
void modify(int p,int l,int r,int ql,int qr,int x) {
if(l>qr||r<ql)
return;
if(l>=ql && r<=qr) {
a[p]=min(a[p],x);
} else {
modify(p*2,l,(l+r)/2,ql,qr,x);
modify(p*2+1,(l+r)/2+1,r,ql,qr,x);
}
}
int query(int p,int l,int r,int pos) {
if(l==r) {
return a[p];
} else {
if(pos<=(l+r)/2) {
return min(query(p*2,l,(l+r)/2,pos), a[p]);
} else {
return min(query(p*2+1,(l+r)/2+1,r,pos), a[p]);
}
}
}
} // namespace seg
int chk(int x)
{
if(x>=0x3f3f3f3f) return -1;
else return x;
}
signed main() {
seg::reset();
scanf("%d%d%d",&n,&m,&q);
for(int i=1; i<=m; i++) {
scanf("%d%d%d",&t1,&t2,&t3);
if(t2==1) t2=t1;
if(t1==n) t1=t2;
e[i] = (edge) {
t1,t2,t3
};
gs.make(t1,t2,t3,i);
gt.make(t2,t1,t3,i);
}
gs.sp(1,n);
gt.sp(n,n);
gs.gen(1,n);
gt.gen(n,1);
int ind = 0;
for(int i=1; gt.fa[i]; i=gt.fa[i]) {
sp.push_back(gt.fid[i]);
use[gt.fid[i]]=++ind;
}
sid[1]=0;
for(int i=0; i<sp.size(); i++) {
sid[e[sp[i]].v]=i+1;
}
int len=sp.size();
for(int i=1; i<=m; i++) {
if(use[i]==0) {
int u=e[i].u, v=e[i].v, w=e[i].w;
seg::modify(1,1,len,sid[gs.c[u]]+1,sid[gt.c[v]],gs.d[u]+gt.d[v]+w);
}
}
for(int i=1; i<=q; i++) {
scanf("%d",&t1);
if(use[t1])
printf("%d\n",chk(seg::query(1,1,len,use[t1])));
else
printf("%d\n",chk(gs.d[n]));
}
}
[HNOI2014] 道路堵塞 - 最短路,线段树的更多相关文章
- BZOJ.3575.[HNOI2014]道路堵塞(最短路 动态SPFA)
题目链接 \(Description\) 给你一张有向图及一条\(1\)到\(n\)的最短路.对这条最短路上的每条边,求删掉这条边后\(1\)到\(n\)的最短路是多少. \(Solution\) 枚 ...
- 洛谷 [HNOI2014]道路堵塞 解题报告
[HNOI2014]道路堵塞 题意 给一个有向图并给出一个这个图的一个\(1\sim n\)最短路,求删去这条最短路上任何一条边后的最短路. 又事SPFA玄学... 有个结论,新的最短路一定是\(1\ ...
- 动态删边SPFA: [HNOI2014]道路堵塞
[HNOI2014]道路堵塞 题目描述 $A$ 国有 $N$座城市,依次标为$1$到$N$.同时,在这$N$座城市间有$M$条单向道路,每条道路的长度是一个正整数.现在,$A$国交通部指定了一条从城市 ...
- BZOJ1018[SHOI2008]堵塞的交通——线段树
题目描述 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个2行C列的矩形网格,网格上的每个点代表一个城市,相邻的城市之间有一条道路,所以总 ...
- Luogu P4246 [SHOI2008]堵塞的交通(线段树+模拟)
P4246 [SHOI2008]堵塞的交通 题意 题目描述 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个\(2\)行\(C\)列的矩形 ...
- HDU5669 Road 分层最短路+线段树建图
分析:(官方题解) 首先考虑暴力,显然可以直接每次O(n^2) 的连边,最后跑一次分层图最短路就行了. 然后我们考虑优化一下这个连边的过程 ,因为都是区间上的操作,所以能够很明显的想到利用线段树来维 ...
- [TJOI2012]桥(最短路+线段树)
有n个岛屿, m座桥,每座桥连通两座岛屿,桥上会有一些敌人,玩家只有消灭了桥上的敌人才能通过,与此同时桥上的敌人会对玩家造成一定伤害.而且会有一个大Boss镇守一座桥,以玩家目前的能力,是不可能通过的 ...
- BZOJ.1018.[SHOI2008]堵塞的交通(线段树维护连通性)
题目链接 只有两行,可能的路径数不多,考虑用线段树维护各种路径的连通性. 每个节点记录luru(left_up->right_up),lurd,ldru,ldrd,luld,rurd,表示这个区 ...
- [BZOJ4699]树上的最短路(最短路+线段树)
https://www.cnblogs.com/Gloid/p/10273902.html 这篇文章已经从头到尾讲的非常清楚了,几乎没有什么需要补充的内容. 首先$O(n\log^2 n)$的做法比较 ...
随机推荐
- vsto 学习
Object到String类型转换的四种方式 通常object到string有四种方式:(假设有object obj) obj.ToString, Convert.ToString, (string) ...
- 剑指offer-面试题38-字符串的排列-全排列
/* 题目: 输入字符串,打印字符串的所有排列. 输入acc,输出[acc, cac, cca]. */ /* 思路: 将字符串看作两部分,第一个字符串和后面的部分. 将第一个字符串与后面字符串依次交 ...
- Python 一键安装全部依赖包
使用 pip requirements.txt 用来记录项目所有的依赖包和版本号,只需要一个简单的 pip 命令就能完成. pip freeze > requirements.txt 生成的文件 ...
- ImportError: libzmq.so.5 报错
https://pkgs.org/download/libzmq.so.5()(64bit) # rpm -ivh zeromq-4.1.4-6.el7.x86_64.rpm
- layui的跳转链接实现分页low
layui.use(['laypage', 'layer'], function(){ var laypage = layui.laypage ,layer = layui.layer; laypag ...
- python3练习100题——049
题目:使用lambda来创建匿名函数. sum=lambda x,y:x+y from functools import reduce reduce(sum,[1,2,3,4,5])
- golang channel 的一次内存错误
起因 原因调查 原因分析 问题解决 总结 起因 今天在做数据库数据读取时, 首先通过多个 goroutine 将从数据库读取的数据写入 channel, 同时通过另一个 goroutine 从 cha ...
- mysql 服务
1.以管理员身份运行cmd:C:\Windows\System32\cmd.exe 右键以管理员身份运行 2.“具体路径” --install “D:\programme installatio ...
- Subway POJ - 2502 spfa
#include<cstdio> #include<cmath> #include<cstring> #include<cstring> #includ ...
- SVM-支持向量机(二)非线性SVM分类
非线性SVM分类 尽管SVM分类器非常高效,并且在很多场景下都非常实用.但是很多数据集并不是可以线性可分的.一个处理非线性数据集的方法是增加更多的特征,例如多项式特征.在某些情况下,这样可以让数据集变 ...