[BJOI2018] 求和 - 树上前缀和,LCA
Posted mollnn
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[BJOI2018] 求和 - 树上前缀和,LCA相关的知识,希望对你有一定的参考价值。
一棵有根树,并且希望多次询问这棵树上一段路径上所有节点深度的 (k) 次方和,而且每次的 (k) 可能是不同的。此处节点深度的定义是这个节点到根的路径上的边数。他把这个问题交给了pupil,但pupil并不会这么复杂的操作,你能帮他解决吗?
Solution
对每个次数,预处理树上前缀和即可
#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N = 300005;
const int mod = 998244353;
vector <int> g[N];
int n,m,t1,t2,t3,fa[N][20],s[N][55],dep[N],vis[N];
void dfs(int p) {
vis[p]=1;
for(int q:g[p]) if(vis[q]==0) {
fa[q][0]=p;
dep[q]=dep[p]+1;
dfs(q);
}
}
void dfs2(int p) {
vis[p]=1;
for(int q:g[p]) if(vis[q]==0) {
for(int k=0;k<=50;k++) s[q][k]+=s[p][k], s[q][k]%=mod;
dfs2(q);
}
}
int lca(int p,int q) {
if(dep[p]<dep[q]) swap(p,q);
for(int i=18;i>=0;--i) if(dep[fa[p][i]]>=dep[q]) p=fa[p][i];
for(int i=18;i>=0;--i) if(fa[p][i]-fa[q][i]) p=fa[p][i],q=fa[q][i];
if(p-q) return fa[q][0];
return p;
}
int dis(int p,int q,int k) {
int l=lca(p,q);
return ((s[p][k]+s[q][k]-s[l][k]-s[fa[l][0]][k])%mod+mod)%mod;
}
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);
}
dfs(1);
for(int i=1;i<=n;i++) {
s[i][0]=1;
for(int j=1;j<=50;j++) {
s[i][j]=s[i][j-1]*dep[i]%mod;
}
}
memset(vis,0,sizeof vis);
dfs2(1);
for(int i=1;i<=18;i++) {
for(int j=1;j<=n;j++) {
fa[j][i]=fa[fa[j][i-1]][i-1];
}
}
cin>>m;
for(int i=1;i<=m;i++) {
cin>>t1>>t2>>t3;
cout<<dis(t1,t2,t3)<<endl;
}
}
以上是关于[BJOI2018] 求和 - 树上前缀和,LCA的主要内容,如果未能解决你的问题,请参考以下文章