P3924 康娜的线段树(线段树+期望)

题目

题 目 传 送 门 \color{red} 题目传送门
小林是个程序媛,不可避免地康娜对这种人类的“魔法”产生了浓厚的兴趣,于是小林开始教她OI。
在这里插入图片描述
今天康娜学习了一种叫做线段树的神奇魔法,这种魔法可以维护一段区间的信息,是非常厉害的东西。康娜试着写了一棵维护区间和的线段树。由于她不会打标记,因此所有的区间加操作她都是暴力修改的。具体的代码如下:

struct Segment_Tree{
#define lson (o<<1)
#define rson (o<<1|1)
    int sumv[N<<2],minv[N<<2];
    inline void pushup(int o){sumv[o]=sumv[lson]+sumv[rson];}
    inline void build(int o,int l,int r){
        if(l==r){sumv[o]=a[l];return;}
        int mid=(l+r)>>1;
        build(lson,l,mid);build(rson,mid+1,r);
        pushup(o);
    }
    inline void change(int o,int l,int r,int q,int v){
        if(l==r){sumv[o]+=v;return;}
        int mid=(l+r)>>1;
        if(q<=mid)change(lson,l,mid,q,v);
        else change(rson,mid+1,r,q,v);
        pushup(o);
    }
}T; 

在修改时,她会这么写:

for(int i=l;i<=r;i++)T.change(1,1,n,i,addv);

显然,这棵线段树每个节点有一个值,为该节点管辖区间的区间和。

康娜是个爱思考的孩子,于是她突然想到了一个问题:

如果每次在线段树区间加操作做完后,从根节点开始等概率的选择一个子节点进入,直到进入叶子结点为止,将一路经过的节点权值累加,最后能得到的期望值是多少?

康娜每次会给你一个值 qwq ,保证你求出的概率乘上 qwq 是一个整数。

这个问题太简单了,以至于聪明的康娜一下子就秒了。

现在她想问问你,您会不会做这个题呢?

输入输出格式
输入格式:
第一行整数 n,m,qwq表示线段树维护的原序列的长度,询问次数,分母。

第二行 n 个数,表示原序列。

接下来 m 行,每行三个数 l,r,x表示对区间[l,r] 加上 x
输出格式:
共 m 行,表示期望的权值和乘上qwq结果。
输入输出样例
输入样例#1:

8 2 1
1 2 3 4 5 6 7 8
1 3 4
1 8 2

输出样例#1:

90
120

题解

  • 朴素的想法:设最终结果为 a n s ans ans,每一层的点权和为 v a l i val_i vali m a x D maxD maxD为最大层数,那么 a n s = ∑ i m a x D v a l i ∗ 1 2 i − 1 ans=\sum_{i}^{maxD}val_i*\frac{1}{2^{i-1}} ans=imaxDvali2i11
  • 然后画个图我们可以发现,从根节点到叶子节点的路径有且只有一条,那么我们可以方便的维护每个叶子节点的期望值,设从根节点到叶子节点的权值和为 s u m i sum_i sumi,每一个节点的深度为 d i d_i di,每个节点的期望为 a n s i ans_i ansi,那么 a n s i = s u m i ∗ 1 2 d i − 1 ans_i=sum_i * \frac{1}{2^{d_i-1}} ansi=sumi2di11
  • 然后我们可以维护一个深度的前缀和我们就可以 O ( 1 ) O(1) O(1)回答询问了。
  • 对于上面的式子 a n s i = s u m i ∗ 1 2 d i − 1 ans_i=sum_i * \frac{1}{2^{d_i-1}} ansi=sumi2di11
  • 每次记录时都除以 2 d i − 1 2^{d_i-1} 2di1非常麻烦,所以我们可以化简一下 a n s i = s u m i ∗ 1 2 d i − 1 = s u m i ∗ 2 m a x D 2 d i − 1 + m a x D = s u m i ∗ 2 m a x D − d i 2 m a x D − 1 ans_i=sum_i * \frac{1}{2^{d_i-1}}=sum_i*\frac{2^{maxD}}{2^{d_i-1+maxD}}=sum_i*\frac{2^{maxD-d_i}}{2^{maxD-1}} ansi=sumi2di11=sumi2di1+maxD2maxD=sumi2maxD12maxDdi
  • 我们在记录时可以直接记录 s u m i ∗ 2 m a x D − d i sum_i*2^{maxD-d_i} sumi2maxDdi,最后直接除以 2 m a x D − 1 2^{maxD-1} 2maxD1即可
  • 对于每次修改来说因为每个点的权值都会改变,设增加的数为 x x x,那么增加的期望值为 x ∗ ∑ i d e p 1 2 i − 1 = x ∗ 2 d e p − 1 2 d e p − 1 = x ∗ ( 2 d e p − 1 ) ∗ 2 m a x D − d e p 2 m a x D − 1 x*\sum_{i}^{dep}\frac{1}{2^{i-1}}=x*\frac{2^{dep} - 1}{2^{dep-1}}=x*\frac{(2^{dep}-1)*2^{maxD-dep}}{2^{maxD-1}} xidep2i11=x2dep12dep1=x2maxD1(2dep1)2maxDdep
  • 可以发现上边两个式子的分母都是 2 m a x D − 1 2^{maxD-1} 2maxD1,我们在统计答案时直接算就好了
  • 其实瞎搞一下就可以了

code

#include <bits/stdc++.h> 
using namespace std; 
typedef long long LL; 
const int maxn = 1e6 + 100; 

template <typename T> 
inline void read(T &s) {
	s = 0; 
	T w = 1, ch = getchar(); 
	while (!isdigit(ch)) { if (ch == '-') w = -1; ch = getchar(); }
	while (isdigit(ch)) { s = (s<<1) + (s<<3) + (ch^48); ch = getchar(); }
	 s *= w;  
}

LL n, m, qwq, maxD, Div, ans;  
LL a[maxn], dep[maxn], sum[maxn]; 
struct node {
	LL l, r; 
	LL sum, add; 
} t[maxn * 4]; 

inline void build(LL p, LL l, LL r, LL d) {
	t[p].l = l, t[p].r = r; 
	if (l == r) {
		t[p].sum = a[l]; 
		dep[l] = d; 
		maxD = max(maxD, d); 
		return ; 
	}
	LL mid = (l + r)>>1; 
	build(p<<1, l, mid, d + 1); 
	build(p<<1|1, mid + 1, r, d + 1); 
	t[p].sum = t[p<<1].sum + t[p<<1|1].sum; 
}

inline LL query(LL p, LL l, LL r, LL d, LL sum) {
	if (l == r) {
		return (1<<d) * (sum + t[p].sum); 
	}
	else {
		int mid = (l + r)>>1; 
		return query(p<<1, l, mid, d - 1, sum + t[p].sum) + query(p<<1|1, mid + 1, r, d - 1, sum + t[p].sum); 
	}
}

inline LL gcd(LL x, LL y) {
	return y ? gcd(y, x % y) : x; 
}

int main() {
	read(n), read(m), read(qwq); 
	for (int i = 1; i <= n; ++i) read(a[i]); 
	build(1, 1, n, 1); 
	ans = query(1, 1, n, maxD - 1, 0); 
	Div = 1<<(maxD - 1); 
	LL g = gcd(qwq, Div); 
	Div /= g; qwq /= g; 
	for (int i = 1; i <= n; ++i) {
		sum[i] = sum[i - 1] + (((1<<dep[i]) - 1) << (maxD - dep[i])); 
	}
	for (int i = 1; i <= m; ++i) {
		LL l, r, x; 
		read(l), read(r), read(x); 
		ans += (sum[r] - sum[l-1]) * x; 
		printf("%lld\n", ans / Div * qwq); 
	}
	return 0; 
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值