链接
题意:给一棵树,求点x,y的最近公共祖先。
#include<bits/stdc++.h>
using namespace std;
#define gcd(a, b) __gcd(a,b)
const long long mod = 1e9 + 7;
const int maxn = 4e4 + 5;
#define int long long
vector<int> v[maxn];
vector<int> w[maxn];
int fa[maxn][31], cost[maxn][31], dep[maxn];
int n, m;
int a, b, c;
void dfs(int u, int k) {
fa[u][0] = k;
dep[u] = dep[k] + 1;
for (int i = 1; i <= 31; i++) {
fa[u][i] = fa[fa[u][i - 1]][i - 1];
cost[u][i] = cost[fa[u][i-1]][i-1]+cost[u][i-1];
}
int len = v[u].size();
for (int i = 0; i < len; ++i) {
if(v[u][i]!=k)
{
cost[v[u][i]][0]=w[u][i];
dfs(v[u][i],u);
}
}
}
int lca(int x, int y) {
if (dep[x] > dep[y]) swap(x, y);
int tmp = dep[y] - dep[x], ans = 0;
for (int j = 0; tmp; ++j, tmp >>= 1)
if (tmp & 1) ans += cost[y][j], y = fa[y][j];
if (y == x) return ans;
for (int j = 30; j >= 0 && y != x; --j) {
if (fa[x][j] != fa[y][j]) {
ans += cost[x][j] + cost[y][j];
x = fa[x][j];
y = fa[y][j];
}
}
ans += cost[x][0] + cost[y][0];
return ans;
}
signed main() {
int t;
cin >> t;
while (t--) {
memset(fa, 0, sizeof(fa));
memset(cost, 0, sizeof(cost));
memset(dep, 0, sizeof(dep));
cin>>n>>m;
for (int i = 1; i < n; ++i) {
scanf("%d %d %d", &a, &b, &c);
v[a].push_back(b);
v[b].push_back(a);
w[a].push_back(c);
w[b].push_back(c);
}
dfs(1, 0);
for (int i = 0; i < m; ++i) {
cin>>a>>b;
printf("%d\n", lca(a, b));
}
}
return 0;
}