题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2125

因为看了TJ又抄了标程,现在感觉还是轻飘飘的……必须再做一遍。

两点间的情况:

1.直到 lca 都没有在一个环上的部分;

2.本来就处在一个环上;

3.本来不在一个环上,快到 lca 的时候开始处在一个环上了。

第一种情况就普通弄就行。处理倍增 lca 数组和根到每个点的最短路dis值。

第二种情况在环上两部分取较短的就行。

第三种情况是前两种的结合。需要找到 p 和 q 刚开始在同一个环上时的那两个进入点 x 和 y。然后dis[ p ] - dis[ x ] + dis[ q ] - dis[ y ]再加上 p、q 环上两部分较短的。

怎么取较短的呢?

可以弄dfs序的距离。

实现的时候第二种和第三种情况可以合并。

细节有一些不明白:环的最高点的深度和环上别的点不一样,也行吗?还有边的数组大小怎么算?

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int N=1e4+;
int n,m,q,hd[N],xnt=,g[N],len[N],cnt,dep[N],f[N][];
int dis[N],ds[N],fa[N],dfn[N],tim;
bool vis[N];
struct Ed{
int nxt,to,w;bool del;
Ed(int n=,int t=,int w=):nxt(n),to(t),w(w) {del=;}
}ed[N<<];
int tabs(int k){return k<?-k:k;}
void add(int x,int y,int z)
{
ed[++xnt]=Ed(hd[x],y,z);hd[x]=xnt;
}
void spfa()
{
memset(dis,0x3f,sizeof dis);dis[]=;
queue<int> q;q.push();vis[]=;
while(q.size())
{
int k=q.front();q.pop();vis[k]=;//don't forget visk=0!!
for(int i=hd[k],v;i;i=ed[i].nxt)
if(dis[v=ed[i].to]>dis[k]+ed[i].w)
{
dis[v]=dis[k]+ed[i].w;
if(!vis[v])vis[v]=,q.push(v);
}
}
}
void circle(int y,int e)
{
int x=ed[e].to;//!
len[++cnt]=ed[e].w;g[y]=cnt;
ed[e].del=ed[e^].del=;//don't forget del!
add(x,y,);add(y,x,);// edw has no limit--only for bfs the dep
// edw is guaranteed by ds[]
for(e=fa[y];(y=ed[e^].to)!=x;e=fa[y])
{
len[cnt]+=ed[e].w;g[y]=cnt;ed[e].del=ed[e^].del=;
add(x,y,);add(y,x,);
}
len[cnt]+=ed[e].w;g[x]=cnt;
}
void dfs(int cr)
{
dfn[cr]=++tim;
for(int i=hd[cr],v;i;i=ed[i].nxt)
if(!dfn[v=ed[i].to])
{
fa[v]=i;ds[v]=ds[cr]+ed[i].w;dfs(v);
}
else if(i!=(fa[cr]^)&&dfn[v]<dfn[cr])circle(cr,i);//cr!!!
}
void bfs()
{
queue<int> q;q.push();dep[]=;
while(q.size())
{
int k=q.front();q.pop();
for(int i=hd[k],v;i;i=ed[i].nxt)
if(!ed[i].del&&!dep[v=ed[i].to])
{
dep[v]=dep[k]+;q.push(v);
f[v][]=k;
for(int j=;j<=;j++)//j not i
{
f[v][j]=f[f[v][j-]][j-];
}
}
}
}
int lca(int x,int y)
{
if(dep[x]<dep[y])swap(x,y);
int p=x,q=y;
for(int i=;i>=;i--)
if(dep[f[x][i]]>=dep[y])x=f[x][i];
if(x==y)return dis[p]-dis[q];
for(int i=;i>=;i--)
if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
if(g[x]!=g[y]||(!g[x]&&!g[y]))return dis[p]+dis[q]-*dis[f[x][]];//&& !g[x]&&!g[y]!!!
int k=tabs(ds[x]-ds[y]);
return dis[p]-dis[x]+dis[q]-dis[y]+min(k,len[g[x]]-k);
// include p&q in different circle but x&y in the same circle;
//because the dep of nodes in the same circle are almost the same
}
int main()
{
scanf("%d%d%d",&n,&m,&q);int x,y,z;
for(int i=;i<=m;i++)
{
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);add(y,x,z);
}
spfa();
dfs();
bfs();
for(int i=;i<=q;i++)
{
scanf("%d%d",&x,&y);
printf("%d\n",lca(x,y));
}
return ;
}

bzoj 2125 最短路——仙人掌两点间最短路的更多相关文章

  1. hdu 2544 最短路(两点间最短路径)

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=2544 方法一:dijkstra算法,求两点之间最短路径. /*********************** ...

  2. bzoj 1023: [SHOI2008]cactus仙人掌图 2125: 最短路 4728: 挪威的森林 静态仙人掌上路径长度的维护系列

    %%% http://immortalco.blog.uoj.ac/blog/1955 一个通用的写法是建树,对每个环建一个新点,去掉环上的边,原先环上每个点到新点连边,边权为点到环根的最短/长路长度 ...

  3. BZOJ.2125.最短路(仙人掌 最短路Dijkstra)

    题目链接 多次询问求仙人掌上两点间的最短路径. 如果是在树上,那么求LCA就可以了. 先做着,看看能不能把它弄成树. 把仙人掌看作一个图(实际上就是),求一遍根节点到每个点的最短路dis[i]. 对于 ...

  4. boost dijkstra获得两点间的最短路

    需求是只需要得到两点间的最短路,不需要求得单源对于全图的最短路,使用boost中的dijsktra_shortest_path,当得到目标点的最短路时直接throw exception. #inclu ...

  5. 【刷题】BZOJ 2125 最短路

    Description 给一个N个点M条边的连通无向图,满足每条边最多属于一个环,有Q组询问,每次询问两点之间的最短路径. Input 输入的第一行包含三个整数,分别表示N和M和Q 下接M行,每行三个 ...

  6. [C++]boost dijkstra获得两点间的最短路

    需求是只需要得到两点间的最短路,不需要求得单源对于全图的最短路,使用boost中的dijsktra_shortest_path,当得到目标点的最短路时直接throw exception. #inclu ...

  7. [CF1051F]The Shortest Statement (LCA+最短路)(给定一张n个点m条有权边的无向联通图,q次询问两点间的最短路)

    题目:给定一张n个点m条有权边的无向联通图,q次询问两点间的最短路 n≤100000,m≤100000,m-n≤20. 首先看到m-n≤20这条限制,我们可以想到是围绕这个20来做这道题. 即如果我们 ...

  8. AOJ GRL_1_C: All Pairs Shortest Path (Floyd-Warshall算法求任意两点间的最短路径)(Bellman-Ford算法判断负圈)

    题目链接:http://judge.u-aizu.ac.jp/onlinejudge/description.jsp?id=GRL_1_C All Pairs Shortest Path Input ...

  9. 用 Excel 测试“绘制两点间连线”的算法

    最近在研究和制作数字示波器,其中涉及一个小算法:需要将 ADC 采样的数值在 TFT LCD 屏幕上面显示并且用“线”连接起来. ADC 按照时序对输入电压采样后,记录的是一个个的数值,如果显示的时候 ...

随机推荐

  1. Python的自省机制

    什么是自省? 在日常生活中,自省(introspection)是一种自我检查行为. 在计算机编程中,自省是指这种能力:检查某些事物以确定它是什么.它知道什么以及它能做什么.自省向程序员提供了极大的灵活 ...

  2. [luogu4255]公主の#18文明游戏

    [luogu4255]公主の#18文明游戏 luogu 发现没有连边,只有删边? 考虑倒着做 开map记M[i][j]表示编号为i的并查集,信仰j的人数 s[i]表示编号为i的并查集的总人数 首先询问 ...

  3. Java & 混型

    1.C++ & 混型 C++能够记住其模板参数的类型,对于混型来说,使用参数化类型更加优雅. #include <string> #include <ctime> #i ...

  4. JVM指令重排

    指令重排的基本原则: a.程序顺序原则:一个线程内保证语义的串行性 b.volatile规则:volatile变量的写,先发生于读 c.锁规则:解锁(unlock)必然发生在随后的加锁(lock)前 ...

  5. python cookbook第三版学习笔记十七:委托属性

    我们想在访问实例的属性时能够将其委托到一个内部持有的对象上,这经常用到代理机制上 class A:     def spam(self,x):         print("class_A: ...

  6. linux c编程:非阻塞I/O

    通常来说,从普通文件读数据,无论你是采用 fscanf,fgets 也好,read 也好,一定会在有限的时间内返回.但是如果你从设备,比如终端(标准输入设备)读数据,只要没有遇到换行符(‘\n’),r ...

  7. git clone了整个远程仓库分支

    git之远程标签下载(远程分支) 一般我们发布一个新版本到线上服务器时都会在版本库中打一个标签,这样我们可以随时查看这个打标签的版本,就是说标签其实是版本库中一个快照.git的标签与分支类似,区别是分 ...

  8. 3.11课·········异常语句与for循环重复

    异常语句try catch finally try//保护执行里面的代码段,若其中一句有错误,直接跳转到catch,不会管下面的内容 { Console.Write("请输入一个整数&quo ...

  9. 数据库 简单查询 Sql Server 学生表 课程表 选课表

    创建教材中的三张表格,并输入相应的数据 Create table student( Sno char(9), Same char(20), Ssex char(2), Sage smallint, S ...

  10. BOM之history

    history是JavaScript中BOM上的一个对象,其中存储了浏览器的历史记录 history存储简单过程 浏览器会将一个窗口中访问的网页进行记录,不管我们通过以下哪种方式改变页面,浏览器都会把 ...