CF607D Power Tree
XLIV.CF607D Power Tree
考虑计算 子树中某个节点 对于 的贡献,发现是 ,其中 是 节点的儿子数。
于是考虑离线,然后建一棵下标为dfs序的线段树,每个节点上记录了区间所有节点对 的贡献。询问 时,就直接查询区间和;但是因为从 到根路径上所有的 都被多算了,因此要除掉。修改时,其父亲的 改变了,就导致父亲子树中所有节点对根的贡献也改变了,但是改变的都是 ,于是直接区间打一个乘法标记即可。
时间复杂度 。
代码:
#include<bits/stdc++.h>
using namespace std;
const int mod=1e9+7;
int ksm(int x,int y=mod-2){int z=1;for(;y;y>>=1,x=1ll*x*x%mod)if(y&1)z=1ll*z*x%mod;return z;}
int n,a[200100],q,sz[200100],dfn[200100],rev[200100],tot,fa[200100],deg[200100];
#define lson x<<1
#define rson x<<1|1
#define mid ((l+r)>>1)
struct SegTree{int sum,tag;SegTree(){sum=0,tag=1;}}seg[800100];
void MUL(int x,int y){seg[x].sum=1ll*seg[x].sum*y%mod,seg[x].tag=1ll*seg[x].tag*y%mod;}
void pushdown(int x){MUL(lson,seg[x].tag),MUL(rson,seg[x].tag),seg[x].tag=1;}
void pushup(int x){seg[x].sum=(seg[lson].sum+seg[rson].sum)%mod;}
void multiply(int x,int l,int r,int L,int R,int val){
if(l>R||r<L)return;
if(L<=l&&r<=R){MUL(x,val);return;}
pushdown(x),multiply(lson,l,mid,L,R,val),multiply(rson,mid+1,r,L,R,val),pushup(x);
}
void turnon(int x,int l,int r,int P){
if(l>P||r<P)return;
if(l==r){seg[x].sum=1ll*a[rev[P]]*seg[x].tag%mod;return;}
pushdown(x),turnon(lson,l,mid,P),turnon(rson,mid+1,r,P),pushup(x);
}
int querytag(int x,int l,int r,int P){
if(l>P||r<P)return 0;
if(l==r)return seg[x].tag;
pushdown(x);return querytag(lson,l,mid,P)+querytag(rson,mid+1,r,P);
}
int querysum(int x,int l,int r,int L,int R){
if(l>R||r<L)return 0;
if(L<=l&&r<=R)return seg[x].sum;
pushdown(x);return (querysum(lson,l,mid,L,R)+querysum(rson,mid+1,r,L,R))%mod;
}
vector<int>v[200100];
int tp[200100],pos[200100];
void dfs(int x){
dfn[x]=++tot,rev[tot]=x,sz[x]=1;
for(auto y:v[x])fa[y]=x,dfs(y),sz[x]+=sz[y];
}
int main(){
scanf("%d%d",&a[n=1],&q);
for(int i=1;i<=q;i++){
scanf("%d%d",&tp[i],&pos[i]);
if(tp[i]==1)scanf("%d",&a[++n]),v[pos[i]].push_back(n),pos[i]=n;
}
for(int i=1;i<=n;i++)deg[i]=1;
dfs(1),turnon(1,1,n,1);
for(int i=1,x;i<=q;i++){
if(tp[i]==1){
x=fa[pos[i]];
multiply(1,1,n,dfn[x],dfn[x]+sz[x]-1,1ll*(deg[x]+1)*ksm(deg[x])%mod),deg[x]++;
turnon(1,1,n,dfn[pos[i]]);
}else x=pos[i],printf("%d\n",1ll*querysum(1,1,n,dfn[x],dfn[x]+sz[x]-1)*(x==1?1:ksm(querytag(1,1,n,dfn[fa[x]])))%mod);
}
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
· 【杂谈】分布式事务——高大上的无用知识?