BZOJ 2725: [Violet 6]故乡的梦
求出最短路径树,对于一个询问(x,y) 若不在树上S->T的链上,则答案不变,若在链上,考虑用一条非树边替换这条边,这条非树边必须跨越x->y这条边,线段树维护区间最小值
#include<cstdio>
#include<algorithm>
#include<queue>
#include<cstring>
#define pr pair<long long,int>
#define mp make_pair
#define sc second
using namespace std;
int cnt,n,m,last[200005],Lv[200005],stack[200005],Fa[200005],vis[200005],pre[200005],Vis[1000005];
long long dis[200005][2],Dep[200005],ANS[200005],tree[800005];
priority_queue<pr,vector<pr>,greater<pr> > q;
struct node{
int to,next;
long long val;
}e[400005];
struct node1{
int x,y,val;
}E[200005];
void add(int a,int b,int c){
e[++cnt].to=b;
e[cnt].next=last[a];
e[cnt].val=c;
last[a]=cnt;
}
void Dijkstra(int S,int cas){
for (int i=1; i<=n; i++) dis[i][cas]=1ll<<60,vis[i]=0;
dis[S][cas]=0;
q.push(mp(0ll,S));
while (!q.empty()){
int x=q.top().sc;
q.pop();
if (vis[x]) continue;
vis[x]=1;
for (int i=last[x]; i; i=e[i].next){
int V=e[i].to;
if (dis[V][cas]>dis[x][cas]+e[i].val){
dis[V][cas]=dis[x][cas]+e[i].val;
if (cas==0) pre[V]=x;
q.push(mp(dis[V][cas],V));
}
}
}
}
void Pre(int x,int fa,long long dep){
Fa[x]=fa,Dep[x]=dep;
for (int i=last[x]; i; i=e[i].next){
int V=e[i].to;
if (V==fa) continue;
Pre(V,x,dep+e[i].val);
}
}
void solve(int x,int lv){
vis[x]=1;
Lv[x]=lv;
for (int i=last[x]; i; i=e[i].next){
int V=e[i].to;
if (vis[V]) continue;
solve(V,lv);
}
}
bool cmp(node a,node b){
return a.to<b.to;
}
void insert(int t,int l,int r,int x,long long key){
if (l==r){
tree[t]=min(tree[t],key);
return;
}
int mid=(l+r)>>1;
if (x<=mid) insert(t<<1,l,mid,x,key);
else insert(t<<1|1,mid+1,r,x,key);
tree[t]=min(tree[t<<1],tree[t<<1|1]);
}
long long query(int t,int l,int r,int x,int y){
if (r<x || l>y) return 1ll<<60;
if (l>=x && r<=y) return tree[t];
int mid=(l+r)>>1;
return min(query(t<<1,l,mid,x,y),query(t<<1|1,mid+1,r,x,y));
}
void build(int t,int l,int r){
if (l==r){
tree[t]=1ll<<60;
return;
}
int mid=(l+r)>>1;
build(t<<1,l,mid);
build(t<<1|1,mid+1,r);
tree[t]=max(tree[t<<1],tree[t<<1|1]);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1; i<=m; i++){
scanf("%d%d%d",&E[i].x,&E[i].y,&E[i].val);
add(E[i].x,E[i].y,E[i].val);
add(E[i].y,E[i].x,E[i].val);
}
int S,T;
scanf("%d%d",&S,&T);
Dijkstra(S,0);
Dijkstra(T,1);
cnt=0;
memset(vis,0,sizeof(vis));
for (int i=1; i<=n; i++) last[i]=0;
for (int i=1; i<=n; i++)
if (pre[i]) {
add(pre[i],i,dis[i][0]-dis[pre[i]][0]);
add(i,pre[i],dis[i][0]-dis[pre[i]][0]);
}
Lv[S]=1;
Pre(S,0,0);
int now=T,top=0;
while (now){
stack[++top]=now;
vis[now]=1;
Vis[now]=top;
now=Fa[now];
}
for (int i=1; i<=top/2; i++) swap(stack[i],stack[top-i+1]);
for (int i=1; i<=top; i++) solve(stack[i],i);
int cnt=0;
for (int i=1; i<=m; i++){
int x=E[i].x,y=E[i].y;
if (Vis[x] && Vis[y] && (Vis[y]==Vis[x]+1 || Vis[x]==Vis[y]+1)) continue;
int X=Lv[x],Y=Lv[y];
if (X>Y) swap(X,Y),swap(x,y);
e[++cnt]=(node){X,Y,dis[x][0]+dis[y][1]+E[i].val};
}
sort(e+1,e+cnt+1,cmp);
build(1,1,top);
int Top=1;
for (int i=1; i<top; i++){
while (Top<=cnt && e[Top].to<=i){
insert(1,1,top,e[Top].next,e[Top].val);
Top++;
}
ANS[i]=query(1,1,top,i+1,top);
}
int q;
scanf("%d",&q);
int cc=0;
while (q--){
cc++;
int x,y;
scanf("%d%d",&x,&y);
if (Vis[x] && Vis[y] && (Vis[x]==Vis[y]+1 || Vis[y]==Vis[x]+1)){
if (dis[T][0]==1ll<<60) printf("Infinity\n");
else{
int X=Lv[x],Y=Lv[y];
if (X>Y) swap(X,Y);
if (ANS[X]!=1ll<<60) printf("%lld\n",ANS[X]);
else printf("Infinity\n");
}
}
else {
if (dis[T][0]!=1ll<<60) printf("%lld\n",dis[T][0]);
else printf("Infinity\n");
}
}
return 0;
}
BZOJ 2725: [Violet 6]故乡的梦的更多相关文章
- BZOJ 2725: [Violet 6]故乡的梦 最短路+线段树
2725: [Violet 6]故乡的梦 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 678 Solved: 204[Submit][Status ...
- [原博客] BZOJ 2725 : [Violet 6]故乡的梦
这个题在bzoj上好像是个权限题,想做的可以去Vani的博客下载测试数据.这里有题面. 简单叙述一下题意:给你一个n个点.m条边的带权无向图,S点和T点,询问Q次删一条给定的边的S-T最短路. 其中 ...
- BZOJ 2725 [Violet 6]故乡的梦 线段树+最短路树
\(\color{#0066ff}{ 题目描述 }\) \(\color{#0066ff}{输入格式}\) \(\color{#0066ff}{输出格式}\) \(\color{#0066ff}{输入 ...
- BZOJ2725 : [Violet 6]故乡的梦
如果S==T,那么答案为0. 如果S与T不连通,那么答案为inf. 否则,S到T的最短路径上至少有一条边. 求出以S为源点的最短路图,是个DAG,随便抓一条S到T的最短路,记为P. 设dpS[x]表示 ...
- 【BZOJ-2725】故乡的梦 Dijsktra + Tarjan + Dinic + BFS + 堆
2725: [Violet 6]故乡的梦 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 502 Solved: 173[Submit][Status ...
- 二分+最短路判定 BZOJ 2709: [Violet 1]迷宫花园
BZOJ 2709: [Violet 1]迷宫花园 Sample Input 5 ######### # # # # # # # #S# # ##### # # ## # # # ### ### ## ...
- [violet6] 故乡的梦
题目 描述 不知每日疲于在城市的水泥森林里奔波的你会不会有时也曾向往过乡村的生活.你会不会幻想过,在夏日一个静谧的午后,你沉睡于乡间路边的树荫里,一片叶子落在了你的肩上, 而你正做着一个悠长的梦,一个 ...
- BZOJ 2716: [Violet 3]天使玩偶
2716: [Violet 3]天使玩偶 Time Limit: 80 Sec Memory Limit: 128 MBSubmit: 1473 Solved: 621[Submit][Statu ...
- BZOJ 2724: [Violet 6]蒲公英
2724: [Violet 6]蒲公英 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 1633 Solved: 563[Submit][Status ...
随机推荐
- python regex
re.match: match from the beginning of the string re.search: scan through the whole string to find a ...
- 转 PHP函数---$_Get()和$_Post()的用法
一.$_Get()和$_Post()函数是用来传值的,即对应两种提交表单的方法,get和post. 二.$_Get方法 (1)获取通过URL的传值 Example 1 新建两个PHP文件,1.php, ...
- 使用Yeoman 创建 angular应用
一.安装 Yeoman npm install yo -g 如果提示当前nodejs版本和npm版本太低,先升级下再安装yeoman. 安装成功后,默认只有webapp和Mocha这两个生成器. 二. ...
- MySQL简单的确定瓶颈
如果接到报警可能需要ssh看看瓶颈是什么,怎么下手 确定os层 确定磁盘是否够用的:df –h 再看看系统整体状态: top 哪些进程占用资源比较多,能杀就杀 系统的负载 vmstat看看wa值,r列 ...
- Spring Security – security none, filters none, access permitAll
1.概述 Spring Security提供了几种将请求模式配置为不安全或允许所有访问的机制.取决于这些机制中的哪一种 - 这可能意味着根本不在该路径上运行安全过滤器链,或者运行过滤器链并允许访问 2 ...
- 《javascript设计模式》笔记之第七章:工厂模式
在读了这章之后,根据我个人现在的理解,工厂模式就是:将一个类或者一个方法称为一个工厂,然后再将一些模块交给这个工厂,让这个工厂按照给它的不同模块产出不同的实例. 下面为正文: 一:简单工厂: 例子: ...
- Java基础之面向对象
面向对象 1.面向对象思想: (1)概述:面向对象是相对于面向过程而言的,面向过程强调的是功能,面向对象强调的是将功能封装进对象,强调具备功能的对象: (2)思想特点: ...
- WebService学习之旅(一)使用JAX-WS发布WebService
JAX-WS全称Java™ API for XML Web Services,是随着JDK1.6及其后续版本发布的方便Java程序员开发WebService应用的一组API,通常简称为JWS,目前版本 ...
- you don't have permission to access / on this server解决
时间:2014-10-13 17:34来源:有何不可 作者:有何不可 举报 点击:56151次 项目部署到Apache Http Server上面,通过apachectl -t 检测配置文件也没有问题 ...
- gcc&g++
原文章 误区一:gcc只能编译c代码,g++只能编译c++代码两者都可以,但是请注意:1.后缀为.c的,gcc把它当作是C程序,而g++当作是c++程序:后缀为.cpp的,两者都会认为是c++程序,注 ...