[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)$的做法比较 ...
随机推荐
- 初识OpenSSH--1
note:保护你的报文(communique)安全 ! 最安全!!! 简介:OpenSSH使用SSH协议进行远程登录的主要连接工具.它对传输数据进行加密,以消除窃听,连接劫持和其他攻击.此外,Ope ...
- clientHeight offsetTop scrollTop
- Pikachu-SSRF(服务器端请求伪造)
SSRF(Server-Side Request Forgery:服务器端请求伪造) 其形成的原因大都是由于服务端提供了从其他服务器应用获取数据的功能,但又没有对目标地址做严格过滤与限制 导致攻击者可 ...
- ASP.NET MVC自定义Numberic属性的验证信息
最近在使用MVC4时碰到一个Model验证的问题:整型属性输入非整型字符串时,错误信息总是“字段 XXX 必须是一个数字”,我总觉得这句话读起来很别扭,所以就萌生了要改变这个默认错误提示信息的念头,但 ...
- Android电源管理基础知识整理
前言 待机.睡眠与休眠的区别? Android开发者官网当中提到"idle states",该如何理解,这个状态会对设备及我们的程序造成何种影响? 进入Doze模式中的idle状态 ...
- spark.streaming.kafka.maxRatePerPartition的理解
spark.streaming.kafka.maxRatePerPartition设定对目标topic每个partition每秒钟拉取的数据条数. 假设此项设为1,批次间隔为10s,目标topic只有 ...
- 【PHP】使用PHP抓取Bing每日图像并为己所用
Bing搜索的首页每天都会推送一张很漂亮的图片,把它保存下来,当做电脑桌面或是自己的网站背景图还不是美滋滋…… 今天的bing图片是这样的 既然要抓取这张图片,首先就得弄清这张图是从何而来的.经过对必 ...
- [CF1303D] Fill The Bag - 贪心
Solution 考虑从低位往高位贪心,设当前在处理第 \(i\) 位,更低位剩余的部分一共可以拼出 \(cnt\) 个 \(2^i\) 如果 \(n\) 的这一位是 \(1\) ,那么这一位就需要处 ...
- 实现字符串和从0到n-1范围内的数字串的一一对应---->poj1200
#include<iostream> using namespace std; ; int num[maxn]; string s; int main() { int nc;//字符串s中 ...
- H5_0020:判断安卓苹果平台
var u = navigator.userAgent, app = navigator.appVersion; var isAndroid = u.indexOf('Android') > - ...