洛谷 P10289 [GESP样题 八级] 小杨的旅游 C++ 完整题解

一、题目链接

P10289 [GESP样题 八级] 小杨的旅游 - 洛谷

二、题目大意

n个节点之间有n - 1条边,其中k个节点是传送门,任意两个传送门之间可以 以0单位地时间相互到达。问从u到v至少需要多少时间?

三、解题思路

输入不必多讲。

cpp 复制代码
cin >> n >> k >> q;
for (int i = 1; i <= n - 1; i++) {
    int x, y;
    cin >> x >> y;
    e[x].push_back(y);
    e[y].push_back(x);
}
for (int i = 1; i <= k; i++) 
    cin >> door[i];

BFS?DFS?

对于这个题目,你肯定会想到用DFS和BFS直接去做。

或者

当然了,更好的方法一定还有,本文只是介绍了一种方法。

正解

分成两种方案:使用传送门和不使用传送门,取快者。

使用传送门

用BFS找到每个节点离最近传送点的距离存入dst数组,结果就是:从起点走到最近传送点 -> 传送到离终点最近的传送点 -> 走到终点 dst[u] + dst[v]

cpp 复制代码
int dst[N]; // dst[i] = i 到 离i最近的传送点 的距离

void bfs() {
    memset(dst, 0x3f, sizeof (dst));
    queue<int> qu;
    for (int i = 1; i <= k; i++) {
        qu.push(door[i]); // 存入所有的传送点
        dst[door[i]] = 0;
    }
    while (!qu.empty()) {
        int now = qu.front();
        qu.pop();
        for (int x : e[now]) {
            if (dst[x] == 0x3f3f3f3f) {
                dst[x] = dst[now] + 1; // 更新x离最近传送点的位置
                qu.push(x);
            }
        }
    }
}
...
{
...
    bfs();
    int res1 = dst[u] + dst[v];
...
}
...

不使用传送门

普通的BFS方法超时了,这里介绍一种可行的方法(LCA)

什么是LCA

LCA的意思是最近公共祖先,如下图,A与B的最近公共祖先是X。

如何用LCA计算A到B的距离
距离dist

首先计算每个节点离根的距离,也就是深度 - 1。

A与B的距离

= dist[A] + dist[B] - 2 * dist[x]

建立dist

cpp 复制代码
int dist[N] = {-1}; // dist[0] = -1 dist[1]才能 = 0
void dfs(int fa, int x) {
    dist[x] = dist[fa] + 1;
    for (int u : e[x]) {
        if (u != fa) // 不能走回头路 省去vis数组
            dfs(x, u);
    }
}
...
{
...
    dfs(0, 1); // 起初给一个无意义的0作为根节点的父亲
...
}
...

接着是lca函数:

cpp 复制代码
int lca(int u, int v);

lca函数中有两个工作要做

  1. 把u和v的深度化作一样 循环上移u或者v,直到dist[u] == dist[v]
cpp 复制代码
// 半伪代码
if (dist[u] > dist[v])
    swap(u, v); // 保证v更深 要不停更新v
while (dist[u] < dist[v]) {
    v = v的父亲;
}
if (u == v) // 如果在没有更新 u 的情况下两者相等 那已经找到了最近公共祖先
    return u;
  1. u和v一起更新 直到两者相等就返回
cpp 复制代码
// 半伪代码
while (u != v) {
    u = u的父亲;
    v = v的父亲;
}
return u;

太慢了!!!

u 或者 v一层一层上去可没时间。

极端情况下就是这样:

每次处理数据都需要进行一次,假设q = 数据最大值,循环次数就是(2 * 100000) ^ 2 = 40,000,000,000。

使用倍增法优化

我们把u,v的第x个祖先看成2 ^ x + 2 ^ y + 2 ^ z...

cpp 复制代码
int f[N][20]; // f[x][i] 节点 x 的 第2^i 个祖先(从下往上)

DFS中记录下任意节点的第2 ^ i个祖先,如下(包含原本的代码)。

cpp 复制代码
void dfs(int fa, int x) {
    dist[x] = dist[fa] + 1;
    f[x][0] = fa; // x的第2^0(1)个祖先就是它爹
    for (int i = 1; i <= 18; i++)
        f[x][i] = f[f[x][i - 1]][i - 1]; // x的第2^i个祖先就是 它2^(i-1)个祖先的第2^(i-1)个祖先 因为2^i = 2^(i-1) + 2^(i-1)
    for (int u : e[x]) {
        if (u != fa)
            dfs(x, u);
    }
}

LCA中也要改动。

cpp 复制代码
int lca(int u, int v) {
    // 1
    if (dist[u] > dist[v])
        swap(u, v);
    for (int i = 18; i >= 0; i--) { // 2^18 > 2*10^5
        if (dist[u] <= dist[f[v][i]])
            v = f[v][i];
        if (u == v) return u;
    }

    //2
    for (int i = 18; i >= 0; i--) {
        if (f[u][i] != f[v][i])
            u = f[u][i], v = f[v][i];
    }
    return f[u][0]; // u和v最后是它们公共祖先的两个儿子 所以它们公共祖先是它们任意一个的父亲
}

块1:i从18开始,试探v的第2^i个祖先 是否 存在 并 深度大于等于u,如果满足以上条件就把v变成v的第2^i个祖先,然后i = 17,16...继续试探,直到u == v或i == 0。

块2:i同样从18开始,试探v的第2^i个祖先 是否和 u的第2^i个祖先 不相等,如果不相等,就更新u和v(同上),循环结束后,u和v一定是它们最近公共祖先的两个儿子, 最后返回它们任意一个的父亲。

四、完整代码

长度有点小小的震撼,但相信你已经看懂了。

cpp 复制代码
#include <iostream>
#include <cstring>
#include <queue>
using namespace std;

const int N = 200005;

vector<int> e[N];
int n, k, q, door[N]; // door[] 存放每个传送点
int dist[N] = {-1}, f[N][20]; // dist[] 每个点离根节点的距离 f[x][i] 节点 x 的 第2^i 个祖先(从下往上)
int dst[N]; // dst[i] = i 到 离i最近的传送点 的距离

void bfs() {
    memset(dst, 0x3f, sizeof (dst));
    queue<int> qu;
    for (int i = 1; i <= k; i++) {
        qu.push(door[i]); // 存入所有的传送点
        dst[door[i]] = 0;
    }
    while (!qu.empty()) {
        int now = qu.front();
        qu.pop();
        for (int x : e[now]) {
            if (dst[x] == 0x3f3f3f3f) {
                dst[x] = dst[now] + 1; // 更新x离最近传送点的位置
                qu.push(x);
            }
        }
    }
}


void dfs(int fa, int x) {
    dist[x] = dist[fa] + 1;
    f[x][0] = fa;
    for (int i = 1; i <= 18; i++)
        f[x][i] = f[f[x][i - 1]][i - 1];
    for (int u : e[x]) {
        if (u != fa)
            dfs(x, u);
    }
}


int lca(int u, int v) {
    if (dist[u] > dist[v])
        swap(u, v);
    for (int i = 18; i >= 0; i--) {
        if (dist[u] <= dist[f[v][i]])
            v = f[v][i];
        if (u == v) return u;
    }

    for (int i = 18; i >= 0; i--) {
        if (f[u][i] != f[v][i])
            u = f[u][i], v = f[v][i];
    }
    return f[u][0];
}


int solve(int u, int v) {
    // ** 使用传送门
    int res1 = dst[u] + dst[v]; // 找离起点最近的传送点 传送至 离终点最近的传送点
    // **不使用传送门
    int res2 = dist[u] + dist[v] - 2 * dist[lca(u, v)];
    return min(res1, res2); // 取小者
}


int main() {
    ios::sync_with_stdio(false);
    cin.tie(0), cout.tie(0);
    cin >> n >> k >> q;
    for (int i = 1; i <= n - 1; i++) {
        int x, y;
        cin >> x >> y;
        e[x].push_back(y);
        e[y].push_back(x);
    }
    for (int i = 1; i <= k; i++) 
        cin >> door[i];
    
    bfs();
    dfs(0, 1);
    while (q--) {
        int u, v;
        cin >> u >> v;
        cout << solve(u, v) << endl;
    }
    return 0;
}
相关推荐
qystca36 分钟前
【16届蓝桥杯寒假刷题营】第2期DAY2
数据结构·c++·算法·深度优先·二分·爆搜
我命由我123452 小时前
游戏引擎 Unity - Unity 设置为简体中文、Unity 创建项目
c语言·开发语言·c++·unity·visualstudio·c#·游戏引擎
南宫生2 小时前
力扣动态规划-19【算法学习day.113】
java·学习·算法·leetcode·动态规划
和风化雨2 小时前
排序算法--插入排序
c语言·c++·算法·排序算法
被AI抢饭碗的人2 小时前
算法题(56):旋转链表
数据结构·算法·链表
暮色初上_3 小时前
700. 二叉搜索树中的搜索
c++·leetcode·回归算法
goomind3 小时前
opencv实现边缘模板匹配
c++·人工智能·opencv·计算机视觉·模板匹配
我一定会有钱3 小时前
C语言:创建带头结点的动态链表:解析与实现
c语言·算法
某个默默无闻奋斗的人3 小时前
二维前缀和:高效求解矩阵区域和问题
java·算法·leetcode·前缀和