【P4320】 道路相遇 (圆方树+LCA)

题目链接
题意:给一张无向图和\(M\)个询问,问\(u,v\)之间的路径的必经之点的个数。

对图建出圆方树,然后必经之点就是两点路径经过的原点个数,用\((dep[u]+dep[v]-dep[LCA]*2)/2+1\)即可算出。

什么你不知道圆方树(说的跟我知道一样)
\(APIO2018\)出来的黑科技,详见\(APIO2018\)铁人两项。

就是对每个点双新建一个点,然后让点双里所有点都对这个点连边。
看图。

#include <cstdio>
const int MAXN = 500010;
const int MAXM = 1000010;
namespace IO{
    int xjc; char ch;
    inline int read(){
        xjc = 0; ch = getchar();
        while(ch < '0' || ch > '9') ch = getchar();
        while(ch >= '0' && ch <= '9'){ xjc = xjc * 10 + ch - '0'; ch = getchar(); }
        return xjc;
    }
}using namespace IO;
inline int min(int a, int b){
    return a > b ? b : a;
}
inline void swap(int &a, int &b){
    xjc = a; a = b; b = xjc;
}
int n, m, k, a, b, cha;
struct Edge{
    int next, to;
};
struct Graph{
    int head[MAXN << 1], num;
    Edge e[MAXM << 1];
    inline void Add(int from, int to){
        e[++num].to = to; e[num].next = head[from]; head[from] = num;
        e[++num].to = from; e[num].next = head[to]; head[to] = num;
    }
}G, T;
int dfn[MAXN], low[MAXN], vis[MAXN], stack[MAXN], f[MAXN << 1][20], dep[MAXN << 1], top, id, cnt;
void Tarjan(int u){
    dfn[u] = low[u] = ++id; stack[++top] = u;
    for(int i = G.head[u]; i; i = G.e[i].next)
        if(!dfn[G.e[i].to]){
            Tarjan(G.e[i].to);
            low[u] = min(low[u], low[G.e[i].to]);
            if(low[G.e[i].to] >= dfn[u]){
                T.Add(u, ++cnt);
                do  T.Add(stack[top], cnt);
                while(stack[top--] != G.e[i].to);
            }
        }
        else low[u] = min(low[u], dfn[G.e[i].to]);
}
void getDF(int u, int fa){
    f[u][0] = fa; dep[u] = dep[fa] + 1;
    for(int i = T. head[u]; i; i = T.e[i].next)
        if(T.e[i].to != fa)
            getDF(T.e[i].to, u);
}
void make_ST(){
    for(int j = 1; j <= 19; ++j)
        for(int i = 1; i <= cnt; ++i)
            f[i][j] = f[f[i][j - 1]][j - 1];
}
inline int LCA(int u, int v){
    if(dep[u] < dep[v]) swap(u, v);
    cha = dep[u] - dep[v];
    if(cha)
        for(int i = 0; i <= 19; ++i)
            if(cha & (1 << i))
                u = f[u][i];
    if(u == v) return u;
    for(int i = 19; ~i; --i)
        if(f[u][i] != f[v][i])
            u = f[u][i], v = f[v][i];
    return f[u][0];
}
int main(){
    cnt = n = read(); m = read();
    for(int i = 1; i <= m; ++i)
        G.Add(read(), read());
    Tarjan(1);
    getDF(1, 0);
    make_ST();
    k = read();
    while(k--){
        a = read(); b = read();
        printf("%d\n", ((dep[a] + dep[b] - (dep[LCA(a, b)] << 1)) >> 1) + 1);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/Qihoo360/p/9808382.html

时间: 2024-10-09 18:22:20

【P4320】 道路相遇 (圆方树+LCA)的相关文章

道路相遇 圆方树

道路相遇 基础圆方树. 不会圆方树看我另一篇文章 LG传送门 发现必经之路上的点一定是简单路径上的点,可以自己手玩.处理无向图上的简单路径,考虑把圆方树建出来,发现答案就是园方树上两点间圆点个数.由于广义园方树上圆方点相间,可以用深度表示答案,发现答案就是\((dep[u] + dep[v] - 2 * dep[lca]) / 2 + 1\). #include <cstdio> #include <cctype> #include <vector> #define R

[SDOI2018]战略游戏 圆方树,树链剖分

[SDOI2018]战略游戏 这题是道路相遇(题解)的升级版,询问的两个点变成了\(S\)个点. LG传送门 还是先建出圆方树,考虑对于询问的\(S\)个点,答案就是圆方树上能包含这些点的最小连通块中的圆点个数减去\(S\).问题变成了怎样求这样的连通块中的圆点个数,直接给结论吧:先搞出树的dfs序,把询问的点按dfs序从小到大排一遍序,每次把答案加上第\(i\)和第\(i + 1\)个点之间的圆点个数,但是不算lca,再加上第\(1\)个和第\(S\)个点之间的圆点个数,然后除以二就得到了这个

【BZOJ】2125: 最短路 圆方树(静态仙人掌)

[题意]给定带边权仙人掌图,Q次询问两点间最短距离.n,m,Q<=10000 [算法]圆方树处理仙人掌问题 [题解]树上的两点间最短路问题,常用倍增求LCA解决,考虑扩展到仙人掌图. 先对仙人掌图建圆方树,圆圆边和原图边权一致.对于每个方点代表的环,记深度最小的点为x,则圆方边的边权是圆点到x的最短距离. 若lca(u,v)为圆点,则两点间最短路转化为圆方树上dis[u]+dis[v]-2*dis[lca].(向上延伸的路径,经过环则必然经过每个方点的x,计算无误) 若lca(u,v)为方点,则

CF487E Tourists 【圆方树 + 树剖 + 堆】

题目链接 CF487E 题解 圆方树 + 树剖 裸题 建好圆方树维护路径上最小值即可 方点的值为其儿子的最小值,这个用堆维护 为什么只维护儿子?因为这样修改点的时候就只需要修改其父亲的堆 这样充分利用了一对一的特性优化了复杂度 如此询问时如果\(lca\)为方点,再询问一下\(lca\)的父亲即可 复杂度\(O(qlog^2n)\) #include<algorithm> #include<iostream> #include<cstring> #include<

Tourists——圆方树

CF487E Tourists 一般图,带修求所有简单路径代价. 简单路径,不能经过同一个点两次,那么每个V-DCC出去就不能再回来了. 所以可以圆方树,然后方点维护一下V-DCC内的最小值. 那么,从任意一个割点进入这个DCC,必然可以绕一圈再从另一个割点出去. 所以,路径上的最小值,就是圆方树路径上的最小值.方点的最小值就是在这个DCC中走一走得到的. 树链剖分+线段树维护路径 用堆维护方点四周的圆点的最小值.然后更新. 一个问题是: 更新一个割点圆点,会影响到四周所有的方点.暴力更新,菊花

CF487E Tourists 圆方树、树链剖分

传送门 注意到我们需要求的是两点之间所有简单路径中最小值的最小值,那么对于一个点双联通分量来说,如果要经过它,则一定会经过这个点双联通分量里权值最小的点 注意:这里不能缩边双联通分量,样例\(2\)就是一个反例 上面这个图如果缩点双会缩成\(3\)个,但是缩边双会将整个图缩成\(1\)个点. 假如我们询问的是\((1,4)\)之间的简单路径,而图中权值最小的点为\(7\)号点,那么如果缩成了边双联通分量,你的答案会是\(7\)号点的权值,意即认为可以走到\(7\)号点,但实际上如果到\(7\)号

[圆方树] Luogu P4630 Duathlon 铁人两项

题目描述 比特镇的路网由 mm 条双向道路连接的 nn 个交叉路口组成. 最近,比特镇获得了一场铁人两项锦标赛的主办权.这场比赛共有两段赛程:选手先完成一段长跑赛程,然后骑自行车完成第二段赛程. 比赛的路线要按照如下方法规划: 先选择三个两两互不相同的路口 s, cs,c和 ff,分别作为比赛的起点.切换点(运动员在长跑到达这个点后,骑自行车前往终点).终点. 选择一条从 ss出发,经过 cc最终到达 ff的路径.考虑到安全因素,选择的路径经过同一个点至多一次. 在规划路径之前,镇长想请你帮忙计

圆方树总结

圆方树:一种将由图转化而成的树,从而大大了增加题目的可解性,且大多广泛用于仙人掌图中. 针对仙人掌图上的圆方树:仙人掌是指一条边至多只被一个环包含的无向图. 树上的点:圆方树上分为两类点,一类是圆点,一类是方点.圆点即原图中所有的点,方点即为了去环而新添加进去的,满足一定性质的点. 构造思路:圆圆边直接加入,对于仙人掌中的任意一个环,每个环上的点在圆方树上对应的圆点向这个环对应的方点连边,方点为一个新建节点. 环的根:指定一个圆点为圆方树的根,把方点的父亲叫做这个方点对应的环的根. 圆方边边权:

圆方树小结

圆方树 jzoj 1914. [2011集训队出题]最短路 这是道圆方树+倍增LCA裸题. 圆方树,顾名思义,就是圆点和方点所组成的树. 而方点就是一个圆的根,一般都是\(dfs\)时第一个到这个圆的那个位置,然后另附一个点当做方点.然后圆所组成的点都连向方点. 而对于这种圆方边的边权,则为它到根的最近值. 从而将一个仙人掌转成了一棵树. 然后对于这棵树,我们就可以用倍增来求出两两点之间的最短路了. 注意的是,对于最后走到的位置,我们要看看它是从上面绕近还是直接从下面走更优!!! 就这样子了.