[CF1304E] 1-Trees and Queries - LCA

由于可以走重边,所以任意一条路径长 + 2 仍然对应至少一条合法路径

很显然我们有 \(3\) 种基本路径

  • 不经过 \((x,y)\)

  • 经过 \(x \to y\)

  • 经过 \(y \to x\)

假设某个基本路径的答案是 \(d\),询问是 \(k\),如果同时满足以下条件,则该基本路径可以作为答案

  • \(d \leq k\)

  • \(d = k \ (mod \ 2)\)

于是暴力做即可

#include <bits/stdc++.h>
using namespace std;

const int N = 200005;
vector <int> g[N];
int n,t1,t2,q,x,y,a,b,k,fa[N][20],dep[N];

void dfs(int p,int fr) {
    for(int q:g[p]) if(q!=fr) {
        fa[q][0]=p;
        dep[q]=dep[p]+1;
        dfs(q,p);
    }
}

int lca(int p,int q) {
    if(dep[p]<dep[q]) swap(p,q);
    for(int i=17;i>=0;--i) if(dep[fa[p][i]]>=dep[q]) p=fa[p][i];
    for(int i=17;i>=0;--i) if(fa[p][i]-fa[q][i]) p=fa[p][i],q=fa[q][i];
    if(p-q) return fa[p][0];
    return p;
}

int dis(int p,int q) {
    return dep[p]+dep[q]-2*dep[lca(p,q)];
}

signed main() {
    ios::sync_with_stdio(false);
    cin>>n;
    for(int i=1;i<n;i++) {
        cin>>t1>>t2;
        g[t1].push_back(t2);
        g[t2].push_back(t1);
    }

    dep[1]=1;
    dfs(1,0);
    for(int i=1;i<=17;i++) {
        for(int j=1;j<=n;j++) {
            fa[j][i]=fa[fa[j][i-1]][i-1];
        }
    }
    cin>>q;
    for(int i=1;i<=q;i++) {
        cin>>x>>y>>a>>b>>k;
        int d;
        d=dis(a,b);
        if(d<=k && (d%2)==(k%2)) {puts("YES"); continue;}
        d=dis(a,x)+1+dis(y,b);
        if(d<=k && (d%2)==(k%2)) {puts("YES"); continue;}
        d=dis(a,y)+1+dis(x,b);
        if(d<=k && (d%2)==(k%2)) {puts("YES"); continue;}
        puts("NO");
    }
}

相关文章:

  • 2021-08-06
  • 2021-11-02
  • 2022-12-23
  • 2022-12-23
  • 2021-12-25
  • 2021-12-24
  • 2021-06-23
  • 2021-09-24
猜你喜欢
  • 2022-03-03
  • 2021-11-25
  • 2021-06-08
  • 2021-06-10
  • 2020-02-16
  • 2022-12-23
  • 2022-03-02
相关资源
相似解决方案