BZOJ 3575 道路堵塞
Description
A国有N座城市,依次标为1到N。同时,在这N座城市间有M条单向道路,每条道路的长度是一个正整数。现在,A国交通部指定了一条从城市1到城市N的路径,并且保证这条路径的长度是所有从城市1到城市N的路径中最短的。不幸的是,因为从城市1到城市N旅行的人越来越多,这条由交通部指定的路径经常发生堵塞。现在A国想知道,这条路径中的任意一条道路无法通行时,由城市1到N的最短路径长度是多少。
Input
输入文件第一行是三个用空格分开的正整数N、M和L,分别表示城市数目、单向道路数目和交通部指定的最短路径包含多少条道路。
按下来M行,每行三个用空格分开的整数a、b和c,表示存在一条由城市a到城市b的长度为c的单向道路。这M行的行号也是对应道路的编号,即其中第1行对应的道路编号为1,第2行对应的道路编号为2,…,第M行对应的道路编号为M。最后一行为L个用空格分开的整数sp(1)…,,sp(L),依次表示从城市1到城市N的由交通部指定的最短路径上的道路的编号。
Output
输出文件包含L行,每行为一个整数,第i行(i=1,2…,,L)的整数表示删去编号为sp(i)的道路后从城市1到城市N的最短路径长度。如果去掉后没有从城市1到城市N的路径,则输出一1。
Sample Input
1 2 1
2 3 1
4 6 1
2 5 2
5 4 3
1 2 3 4
Sample Output
7
7
-1
HINT
100%的数据满足2<N<100000,1<M<200000。所用道路长度大于0小于10000。
这题我用的ydc的做法。。。但是不得不承认他还是太屌了。。。QAQ
首先很容易想到即使去掉一条边之后,最短路依然是pre[a]+suf[b]+dis(a,b),其中a,b均为最短路上的点,pre指a到起点的最短路距离,b为b到终点的最短路距离,dis(a,b)为a到b的距离(不走最短路上的边)。。。
然后我就想到这里了。。。
之后就是ydc神奇做法。首先肯定还是枚举删去的边,然后从这条边的出发点开始跑spfa,不走删掉的边(注意,不要取清空dis数组,因为每个点的dis肯定是单调递减的,不然必然会TLE)。从该点跑到另一个最短路上的点p(p必须在所删的边之后),然后把整条路径的长度和p一起加入平衡树中。出来平衡树中取出长度最小的边,若其做对应的p点(最短路)并没有在所删的边之后,该边删除。若平衡树为空,则输出-1。
不得不说这个做法很漂亮,它利用到了枚举边的单调性,反正我肯定想不到。。。。
如果实在是不懂的话,看代码,代码应该很好理解。
#include<set>
#include<queue>
#include<cstring>
#include<cstdio>
#include<cstdlib>
using namespace std; #define maxn (200010)
int side[maxn],toit[maxn],next[maxn],dis[maxn],pos[maxn],occ[maxn],stack[maxn];
int pre[maxn],suf[maxn],len[maxn],n,m,l,id,cnt,p[maxn],edge[maxn],val[maxn];
bool ban[maxn],in[maxn];
struct node
{
int to,val;
friend inline bool operator <(const node &a,const node &b)
{
if (a.val != b.val) return a.val < b.val;
else return pos[a.to] < pos[b.to];
}
};
multiset <node> MS; inline int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
} inline void add(int a,int b,int c)
{
next[++cnt] = side[a]; side[a] = cnt;
toit[cnt] = b; len[cnt] = c;
} inline void spfa(int d,int source,int lim)
{
queue <int> team; int top = ;
team.push(source); dis[source] = d; ++id;
while (!team.empty())
{
int now = team.front(); team.pop();
for (int i = side[now];i;i = next[i])
{
if (ban[i]) continue;
if (pos[toit[i]] >= lim)
{
if (occ[toit[i]] != id)
occ[toit[i]] = id,stack[++top] = toit[i],val[toit[i]] = dis[now]+len[i]+suf[pos[toit[i]]];
else val[toit[i]] = min(val[toit[i]],dis[now]+len[i]+suf[pos[toit[i]]]);
}
else if (dis[toit[i]] > dis[now]+len[i])
{
dis[toit[i]] = dis[now] + len[i];
if (!in[toit[i]]) in[toit[i]] = true,team.push(toit[i]);
}
}
in[now] = false;
}
for (int i = ;i <= top;++i) MS.insert((node) {stack[i],val[stack[i]]});
} inline void work()
{
pos[] = p[] = ;
for (int i = ;i <= l;++i)
edge[i] = read(),pos[toit[edge[i]]] = i+,p[i+] = toit[edge[i]];
for (int i = ;i <= l;++i) pre[i] = pre[i-]+len[edge[i]];
for (int i = l;i;--i) suf[i] = suf[i+]+len[edge[i]];
memset(dis,0x7,*(n+)); dis[] = ;
for (int i = ;i <= l;++i)
{
ban[edge[i]] = true; spfa(pre[i-],p[i],i+); ban[edge[i]] = false;
while (!MS.empty()&&pos[MS.begin()->to]<=i) MS.erase(MS.begin());
if (MS.empty()) puts("-1");
else printf("%d\n",MS.begin()->val);
}
} int main()
{
freopen("3575.in","r",stdin);
freopen("3575.out","w",stdout);
n = read(),m = read(),l = read(); int a,b,c;
while (m--) a = read(),b = read(),c = read(),add(a,b,c);
work();
fclose(stdin); fclose(stdout);
return ;
}
BZOJ 3575 道路堵塞的更多相关文章
- 洛谷 [HNOI2014]道路堵塞 解题报告
[HNOI2014]道路堵塞 题意 给一个有向图并给出一个这个图的一个\(1\sim n\)最短路,求删去这条最短路上任何一条边后的最短路. 又事SPFA玄学... 有个结论,新的最短路一定是\(1\ ...
- 动态删边SPFA: [HNOI2014]道路堵塞
[HNOI2014]道路堵塞 题目描述 $A$ 国有 $N$座城市,依次标为$1$到$N$.同时,在这$N$座城市间有$M$条单向道路,每条道路的长度是一个正整数.现在,$A$国交通部指定了一条从城市 ...
- BZOJ 2200 道路与航线(图论)
BZOJ 2200 道路与航线 题目大意 有两种路,一种没负数,一种没环,求单源最短路. solution 存在负边权Dij一定不能用嘛,显然不是 根据题意能看出来是tarjan,将双向边缩点,得到的 ...
- bzoj 3575: [Hnoi2014]道路堵塞
Description A 国有N座城市,依次标为1到N.同时,在这N座城市间有M条单向道路,每条道路的长度是一个正整数.现在,A国交通部指定了一条从城市1到城市N的路径, 并且保证这条路径的长度是所 ...
- 【BZOJ】3575: [Hnoi2014]道路堵塞
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3575 大概的做法是,按照顺序枚举每一条要删去的边,(假设当前点为$u$,在最短路径上的下一 ...
- BZOJ.3575.[HNOI2014]道路堵塞(最短路 动态SPFA)
题目链接 \(Description\) 给你一张有向图及一条\(1\)到\(n\)的最短路.对这条最短路上的每条边,求删掉这条边后\(1\)到\(n\)的最短路是多少. \(Solution\) 枚 ...
- 数据结构(线段树):BZOJ 1018: [SHOI2008]堵塞的交通traffic
1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec Memory Limit: 162 MBSubmit: 2638 Solved: 864 Descri ...
- BZOJ 1018 [SHOI2008]堵塞的交通traffic
1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec Memory Limit: 162 MBSubmit: 2247 Solved: 706[Submit ...
- BZOJ 1018: [SHOI2008]堵塞的交通traffic [线段树 区间信息]
1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec Memory Limit: 162 MBSubmit: 3064 Solved: 1027[Submi ...
随机推荐
- JAVA Web学习篇--Servlet
Servlet由来 做过BS项目的人都知道,浏览器可以依据HTML静态标记语言来显示各式各样的网页.可是假设我们须要在网页上完毕一些业务逻辑:比方登陆验证.或者说网页显示的内容在server的数据库中 ...
- hdoj Last non-zero Digit in N! 【数论】
找规律! 求N!最后非0位的值.比方2是120的最后一个不是0的值. 输入N比較大,要大数保存. 注意到最后0的个数是与5的因数的个数相等.设f(n)为n!的最后非0位. 那么f(n)=((n%5)! ...
- 手机相机ISO是什么
要说什么是ISO还要从传统胶片相机说起,ISO被 称为感光度,它是衡量传统相机所使用胶片感光速度的国际统一指标,其数值反映了胶片感光时的速度(其实是银元素与光线的光化学反应速率).而对于现在并不 使用 ...
- SVNclient安装与使用
Technorati 标签: SVN SVNclient安装与使用 1 下载最新版本号1.5.2 最新版本号:TortoiseSVN-1.5.2.13595-win32-svn-1.5.1.msi 下 ...
- MySQL——修改root密码的4种方法(以windows为例)
方法1: 用SET PASSWORD命令 首先登录MySQL. 格式:mysql> set password for 用户名@localhost = password('新密码'); 例子:my ...
- 自己写的demo。List<HashMap<String,Object>>=new ArrayList<HashMap<String,Object>>
package com.pb.collection; import java.util.ArrayList; import java.util.HashMap; import java.util.It ...
- Active Desktop--桌面字体背景被修改
怎么修改回来 步骤如下 方法一.在桌面上点击右键 -- 排列图标 -- 去掉“在桌面上锁定Web项目”上的勾. 方法二.右键点击我的电脑 -- 属性 -- 高级 -- 点击“性能”下面的“设置”按钮, ...
- Unity3D 5.0简单的射线检测实现跳跃功能
这里是一个简单的跳跃,5.0和其他版本貌似不一样,并且,再起跳功能做的不完全. 不过一个基本的思路在这里. 1.首先,射线检测,这里是利用一个空对象,放到主角对象的下面 2.然后调节射线的位置,在主角 ...
- 我的网站终于连续一星期,ip数1000以上了
相信每个程序员都有一个站长梦,我也是如此.说来惭愧,从2015年4月份开始,我投入了50块的域名费用,50块的空间费用,现在才回本. 由于终于从百度联盟的手里赚了一百块,按耐不住的菜鸟之冲动,于是决定 ...
- Android之获取本地图片并压缩方法
这两天在做项目时,做到上传图片功能一块时,碰到两个问题,一个是如何获取所选图片的路径,一个是如何压缩图片,在查了一些资料和看了别人写的后总算折腾出来了,在此记录一下. 首先既然要选择图片,我们就先要获 ...