[BJOI2018]求和

题目大意:
  一个$n(n\le3\times10^5)$个结点的有根树。$m(m\le3\times10^5)$次询问,每次询问$x,y$路径上所有结点深度$k(k\le50)$次方和。

思路:
  树上前缀和+LCA即可。

 1 #include<cstdio>
 2 #include<cctype>
 3 #include<algorithm>
 4 #include<sys/mman.h>
 5 #include<sys/stat.h>
 6 typedef long long int64;
 7 class MMapInput {
 8     private:
 9         char *buf,*p;
10         int size;
11     public:
12         MMapInput() {
13             register int fd=fileno(stdin);
14             struct stat sb;
15             fstat(fd,&sb);
16             size=sb.st_size;
17             buf=reinterpret_cast<char*>(mmap(0,size,PROT_READ,MAP_PRIVATE,fileno(stdin),0));
18             p=buf;
19         }
20         char getchar() {
21             return (p==buf+size||*p==EOF)?EOF:*p++;
22         }
23 };
24 MMapInput mmi;
25 inline int getint() {
26     register char ch;
27     while(!isdigit(ch=mmi.getchar()));
28     register int x=ch^'0';
29     while(isdigit(ch=mmi.getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
30     return x;
31 }
32 const int N=3e5+1,logN=19,K=51,mod=998244353;
33 int h[N];
34 struct Edge {
35     int to,next;
36 };
37 Edge e[N<<1];
38 inline void add_edge(const int &u,const int &v) {
39     e[++h[0]]=(Edge){v,h[u]};h[u]=h[0];
40     e[++h[0]]=(Edge){u,h[v]};h[v]=h[0];
41 }
42 int dep[N]={-1},anc[N][logN],sum[N][K];
43 inline int log2(const float &x) {
44     return ((unsigned&)x>>23&255)-127;
45 }
46 void dfs(const int &x,const int &par) {
47     dep[x]=dep[anc[x][0]=par]+1;
48     for(register int i=1,v=dep[x];i<K;i++) {
49         sum[x][i]=(sum[par][i]+v)%mod;
50         v=(int64)v*dep[x]%mod;
51     }
52     for(register int i=1;i<=log2(dep[x]);i++) {
53         anc[x][i]=anc[anc[x][i-1]][i-1];
54     }
55     for(int i=h[x];i;i=e[i].next) {
56         const int &y=e[i].to;
57         if(y==par) continue;
58         dfs(y,x);
59     }
60 }
61 inline int lca(int x,int y) {
62     if(dep[x]!=dep[y]) {
63         if(dep[x]<dep[y]) std::swap(x,y);
64         for(register int i=log2(dep[x]-dep[y]);~i;i--) {
65             if(dep[anc[x][i]]>=dep[y]) x=anc[x][i];
66         }
67     }
68     if(x==y) return x;
69     for(register int i=log2(dep[x]);~i;i--) {
70         if(anc[x][i]!=anc[y][i]) {
71             x=anc[x][i];
72             y=anc[y][i];
73         }
74     }
75     return anc[x][0];
76 }
77 inline int query(const int &x,const int &y,const int &k) {
78     const int z=lca(x,y);
79     return ((sum[x][k]+sum[y][k]-sum[z][k]-sum[anc[z][0]][k])%mod+mod)%mod;
80 }
81 int main() {
82     const int n=getint();
83     for(register int i=1;i<n;i++) {
84         add_edge(getint(),getint());
85     }
86     dfs(1,0); 
87     for(register int i=getint();i;i--) {
88         const int x=getint(),y=getint(),k=getint();
89         printf("%d\n",query(x,y,k));
90     }
91     return 0;
92 }

 

posted @ 2018-04-17 15:43  skylee03  阅读(315)  评论(0编辑  收藏  举报