Descriptioin

浮生有梦三千场

穷尽千里诗酒荒

徒把理想倾倒

不如早还乡

温一壶风尘的酒

独饮往事迢迢

举杯轻思量

泪如潮青丝留他方

——乌糟兽/愚青《旧词》

你已经解决了五个问题,不妨在这大树之下,吟唱旧词一首抒怀。最后的问题就是关于这棵树的,它的描述很简单。

给定一棵 \(n\) 个点的有根树,节点标号 \(1 \sim n\),11 号节点为根。

给定常数 \(k\) 。

给定 \(Q\) 个询问,每次询问给定 \(x,y\)。

求:

\(\sum\limits_{i\leq x} depth(lca(i,y))^k\)

\(lca(x,y)\) 表示节点 \(x\) 与节点 \(y\) 在有根树上的最近公共祖先。

\(depth(x)\) 表示节点 \(x\) 的深度,根节点的深度为 1。

由于答案可能很大,你只需要输出答案模 998244353 的结果。

Input

输入包含 \(n+Q\) 行。

第 1 行,三个正整数 \(n,Q,k\) 。

第 \(i=2 \sim n\) 行,每行有一个正整数 \(f_i,(1 \leq f_i \leq n)\),表示编号为 \(i\) 的节点的父亲节点的编号。

接下来 $ Q$ 行,每行两个正整数 \(x,y(1 \leq x,y \leq n)\),表示一次询问。

Output

输出包含 \(Q\) 行,每行一个整数,表示答案模 998244353 的结果。

Sample Input

5 5 2

1

4

1

2

4 3

5 4

2 5

1 2

3 2

Sample Output

15

11

5

1

6

HINT

样例解释

输入的树:

每个点的深度分别为 1,2,3,2,3。

第一个询问 \(x = 4,y = 3\),容易求出:

\(lca(1,3)=1,lca(2,3)=1,lca(3,3)=3,lca(4,3)=4\)

于是 \(depth(1) +depth(1) +depth(3) +depth(4) =1+1+9+4=15\) 。

数据范围 \(n\leq 50000,m\leq 50000,1\leq k \leq 10^9\)


想法

首先一句题外话,我真的很喜欢这种有些文学范儿的题~

言归正传!这个题跟 \([LNOI2014]LCA\) 很像,挺套路的。

离线,把所有询问按 \(x\) 从小到大排序后处理。

考虑当 \(k=1\) 时,将 \(1\leadsto [1,x]\) 每条路径上经过的点的值都加1,统计 \(1\leadsto y\) 路径上所有点的值之和就是答案

这样做的原理为 对于任意深度为 \(i\) 的点,它对答案的贡献为 $ i1-(i-1)1=1 $

那么当 \(k \neq 1\) 时也同理,任意深度为 \(i\) 的点对答案的贡献为 \(i^k-(i-1)^k=c\) ,路径上若经过这个点,则该点的值加 \(c\)

由于所有点深度固定,\(k\) 也固定, \(c\) 是很容易处理出来的。

至于如何修改与统计,那当然是树链剖分+线段树咯!(虽说 \(lct\) 也行,但想想那代码量与极大的常数还是放弃吧 )


代码

#include<cstdio>
#include<iostream>
#include<algorithm> using namespace std; int read(){
int x=0;
char ch=getchar();
while(!isdigit(ch)) ch=getchar();
while(isdigit(ch)) x=x*10+ch-'0',ch=getchar();
return x;
} const int N = 50005;
const int xzy = 998244353;
typedef long long ll; struct node{
int v;
node *nxt;
}pool[N],*h[N];
int cnt;
void addedge(int u,int v){
node *p=&pool[++cnt];
p->v=v;p->nxt=h[u];h[u]=p;
} int fa[N],son[N],dep[N],size[N];
void dfs1(int u){
int v,sonnum=0;
size[u]=1;
for(node *p=h[u];p;p=p->nxt){
dep[v=p->v]=dep[u]+1;
dfs1(v);
if(size[v]>sonnum) sonnum=size[v],son[u]=v;
size[u]+=size[v];
}
}
int top[N],w[N],rk[N],tot;
void dfs2(int u){
int v=son[u];
if(v){
top[v]=top[u];
w[v]=++tot;
rk[tot]=v;
dfs2(v);
}
for(node *p=h[u];p;p=p->nxt)
if((v=p->v)!=son[u]){
top[v]=v;
w[v]=++tot;
rk[tot]=v;
dfs2(v);
}
} int n,m,k;
int Pow_mod(int x,int y){
int ret=1;
while(y){
if(y&1) ret=((ll)ret*x)%xzy;
x=((ll)x*x)%xzy;
y>>=1;
}
return ret;
} struct tree{
tree *ch[2];
int e,sum,lazy;
}pool2[N*2],*root;
int cnt2;
void build(tree *p,int l,int r){
p->sum=p->lazy=0;
if(l==r){
p->e=(Pow_mod(dep[rk[l]],k)-Pow_mod(dep[rk[l]]-1,k)+xzy)%xzy;
return;
}
int mid=(l+r)>>1;
build(p->ch[0]=&pool2[++cnt2],l,mid);
build(p->ch[1]=&pool2[++cnt2],mid+1,r);
p->e=((ll)p->ch[0]->e+p->ch[1]->e)%xzy;
}
void update(tree *p) { p->sum=((ll)p->ch[0]->sum+p->ch[1]->sum)%xzy; }
void pushdown(tree *p){
if(!p->lazy) return;
for(int i=0;i<2;i++){
(p->ch[i]->lazy+=p->lazy)%=xzy;
p->ch[i]->sum=((ll)p->ch[i]->sum+1ll*p->lazy*p->ch[i]->e%xzy)%xzy;
}
p->lazy=0;
}
void change(tree *p,int l,int r,int L,int R){
if(l==L && r==R){
p->lazy++;
p->sum=(p->sum+p->e)%xzy;
return;
}
pushdown(p);
int mid=(l+r)>>1;
if(R<=mid) change(p->ch[0],l,mid,L,R);
else if(L>mid) change(p->ch[1],mid+1,r,L,R);
else{
change(p->ch[0],l,mid,L,mid);
change(p->ch[1],mid+1,r,mid+1,R);
}
update(p);
}
int query(tree *p,int l,int r,int L,int R){
if(l==L && r==R) return p->sum;
pushdown(p);
int mid=(l+r)>>1;
if(R<=mid) return query(p->ch[0],l,mid,L,R);
else if(L>mid) return query(p->ch[1],mid+1,r,L,R);
return (query(p->ch[0],l,mid,L,mid)+query(p->ch[1],mid+1,r,mid+1,R))%xzy;
} void add(int x){
while(x){
change(root,1,n,w[top[x]],w[x]);
x=fa[top[x]];
}
}
int ask(int x){
int ret=0;
while(x){
(ret+=query(root,1,n,w[top[x]],w[x]))%=xzy;
x=fa[top[x]];
}
return ret;
} struct data{
int x,y,id;
bool operator < (const data &b) const{ return x<b.x; }
}d[N];
int ans[N]; int main()
{
n=read(); m=read(); k=read();
for(int i=2;i<=n;i++) {
fa[i]=read();
addedge(fa[i],i);
}
for(int i=0;i<m;i++) {
d[i].x=read(); d[i].y=read();
d[i].id=i;
}
sort(d,d+m); dep[1]=1;
dfs1(1);
top[1]=1; w[1]=++tot; rk[tot]=1;
dfs2(1);
build(root=&pool2[++cnt2],1,n); int t=0;
for(int i=0;i<m;i++){
while(t<d[i].x) add(++t);
ans[d[i].id]=ask(d[i].y);
}
for(int i=0;i<m;i++) printf("%d\n",ans[i]); return 0;
}

[bzoj5507] [洛谷P5305] [gzoi2019]旧词的更多相关文章

  1. 【BZOJ5507】[GXOI/GZOI2019]旧词(树链剖分,线段树)

    [BZOJ5507][GXOI/GZOI2019]旧词(树链剖分,线段树) 题面 BZOJ 洛谷 题解 如果\(k=1\)就是链并裸题了... 其实\(k>1\)发现还是可以用类似链并的思想,这 ...

  2. P5305 [GXOI/GZOI2019]旧词

    题目地址:P5305 [GXOI/GZOI2019]旧词 这里是官方题解 \[\sum_{i \leq x}^{}\ depth(lca(i,y))^k\] \(k = 1\) 求的是 \(\sum_ ...

  3. [LOJ3088][GXOI/GZOI2019]旧词——树链剖分+线段树

    题目链接: [GXOI/GZOI2019]旧词 对于$k=1$的情况,可以参见[LNOI2014]LCA,将询问离线然后从$1$号点开始对这个点到根的路径链修改,每次询问就是对询问点到根路径链查询即可 ...

  4. BZOJ5507 GXOI/GZOI2019旧词 (树链剖分+线段树)

    https://www.cnblogs.com/Gloid/p/9412357.html差分一下是一样的问题.感觉几年没写过树剖了. #include<iostream> #include ...

  5. luogu P5305 [GXOI/GZOI2019]旧词

    传送门 先考虑\(k=1\),一个点的深度就是到根节点的路径上的点的个数,所以\(lca(x,y)\)的深度就是\(x\)和\(y\)到根路径的交集路径上的点的个数,那么对于一个询问,我们可以对每个点 ...

  6. P5305-[GXOI/GZOI2019]旧词【树链剖分,线段树】

    正题 题目链接:https://www.luogu.com.cn/problem/P5305 题目大意 给一棵有根树和\(k\),\(Q\)次询问给出\(x,y\)求 \[\sum_{i=1}^{x} ...

  7. [GXOI/GZOI2019]旧词(树上差分+树剖)

    前置芝士:[LNOI2014]LCA 要是这题放HNOI就好了 原题:\(\sum_{l≤i≤r}dep[LCA(i,z)]\) 这题:\(\sum_{i≤r}dep[LCA(i,z)]^k\) 对于 ...

  8. [GXOI/GZOI2019]旧词

    很像LNOI 2014 LCA那道题. 同样的套路,离线以后直接扫描线. k=1的话就是原题. 考虑一般情况. 原本的做法是对x到根的这条链做一下区间+1操作,目的是为了是的在深度为i的位置得到的贡献 ...

  9. BZOJ 5507: [gzoi2019]旧词 LCT

    和之前那个 [LNOI]LCA 几乎是同一道题,就是用动态树来维护查分就行. code: #include <bits/stdc++.h> using namespace std; #de ...

随机推荐

  1. linux 内核定时器

    无论何时你需要调度一个动作以后发生, 而不阻塞当前进程直到到时, 内核定时器是给你 的工具. 这些定时器用来调度一个函数在将来一个特定的时间执行, 基于时钟嘀哒, 并且 可用作各类任务; 例如, 当硬 ...

  2. js 的this指向问题

    this指向的,永远只可能是对象! this指向谁,永远不取决于this写在哪!而是取决于函数在哪调用. this指向的对象,我们称之为函数的上下文context,也叫函数的调用者. 1:通过函数名直 ...

  3. CSS3 属性学习

    fill-available表示撑满可用空间(包括高度,宽度)[此处包括padding和margin会尽可能的撑满,只对于行内块(inline-block)和块元素(block)起作用,webkit内 ...

  4. 使用Squid做代理服务器,Squid单网卡透明代理配置详解(转)

    使用Squid做代理服务器 说到代理服务器,我们最先想到的可能是一些专门的代理服务器网站,某些情况下,通过它们能加快访问互联网的速度.其实,在需要访问外部的局域网中,我们自己就能设置代理,把访问次数较 ...

  5. 使用FluentEmail发送outlook邮件

    一,邮箱账号相关设置 1,创建outLook邮箱. 2,进入邮箱设置->同步电子邮件->允许设备和应用使用pop 3,设置microsoft账号的应用程序密码->进入安全性页面-&g ...

  6. Java方法参数:

    一个方法不能修改一个基本数据类型的参数 一个方法可以改变一个对象参数的状态 一个方法不能实现让对象参数引用一个新的对象 案例1: 一个方法不能修改一个基本数据类型的参数 String a = &quo ...

  7. Ambari+HDP+HDF离线安装包下载清单

    Ambari 2.7.3 Repositories OS Format URL RedHat 7 CentOS 7 Oracle Linux 7 Base URL http://public-repo ...

  8. [梁山好汉说IT] 以水浒传为例讲解贝叶斯定理

    0x00 摘要 看看呼延灼如何利用贝叶斯定理来判断 "自己是否是公明哥哥的心腹". 0x01 IT概念 1. 贝叶斯定理 贝叶斯定理是用来解决"逆概率"问题的, ...

  9. redis 本地安装

    1.redis介绍 Redis是有名的NoSql数据库,一般Linux都会默认支持.但在Windows环境中,可能需要手动安装设置才能有效使用.简单介绍一下Windows下Redis服务的安装方法. ...

  10. 轻松搭建基于 SpringBoot + Vue 的 Web 商城应用

    背景介绍 首先介绍下在本文出现的几个比较重要的概念: 函数计算(Function Compute): 函数计算是一个事件驱动的服务,通过函数计算,用户无需管理服务器等运行情况,只需编写代码并上传.函数 ...