可以发现这个写挂的树状数组求的是后缀和。find(r)-find(l-1)在模2意义下实际上查询的是l-1~r-1的和,而本来要查询的是l~r的和。也就是说,若结果正确,则a[l-1]=a[r](mod 2)。

  一个很容易想到的思路是线段树维护每一位为1的概率。然而这其实是不对的,因为每一位是否为1并非独立事件。

  世界上没有什么事情是用一维线段树解决不了的,如果有,那就两维

  我们维护每两位之间相同的概率。考虑一次操作对某两位的影响。若该次操作包含两位中的x位,那么改变两者间相同状态的概率就是x/len,len为该次修改的区间长度。设原相同概率为pi,j,那么操作后概率就变为(1-pi,j)*x/len+pi,j*(1-x/len)。这个式子神奇的满足交换律结合律,算的时候我们就不用管顺序了。

  用二维线段树就可以维护了。修改时先拆成外层线段树上的区间再在内层线段树修改,查询时将所有外层区间包含该点的内层线段树上的操作合在一起。并且需要标记永久化。

  注意l=1时find(l-1)直接返回0而不是整个数组的和,此时询问的是r后缀和r前缀是否相等,需要特判一下,记录修改了几次以及r这一位为0的概率(即和第0位相同的概率)。

  (写起来出乎意料的短

  (空间需要开的非常大

  (当然也可以cdq分治变成静态数点扫描线扫过去就好了

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
#define N 100010
#define P 998244353
int n,m,tot=,inv[N],L[N<<],R[N<<],cnt=;
int root[N<<];
struct data{int l,r,x;
}tree[N*];
void inc(int &x,int y){x+=y;if (x>=P) x-=P;}
int calc(int x,int y){return (1ll*x*(P+-y)+1ll*y*(P+-x))%P;}
void build(int k,int l,int r)
{
L[k]=l,R[k]=r;
if (l==r) return;
int mid=l+r>>;
build(k<<,l,mid);
build(k<<|,mid+,r);
}
void modify(int &k,int l,int r,int x,int y,int a)
{
if (!k) k=++cnt;
if (l==x&&r==y)
{
tree[k].x=calc(tree[k].x,a);
return;
}
int mid=l+r>>;
if (y<=mid) modify(tree[k].l,l,mid,x,y,a);
else if (x>mid) modify(tree[k].r,mid+,r,x,y,a);
else modify(tree[k].l,l,mid,x,mid,a),modify(tree[k].r,mid+,r,mid+,y,a);
}
int query(int k,int l,int r,int x,int ans)
{
if (!k) return ans;
ans=calc(ans,tree[k].x);
if (l==r) return ans;
int mid=l+r>>;
if (x<=mid) return query(tree[k].l,l,mid,x,ans);
else return query(tree[k].r,mid+,r,x,ans);
}
void change(int k,int l,int r,int x,int y,int a)
{
if (L[k]==l&&R[k]==r) {modify(root[k],,n,x,y,a);return;}
int mid=L[k]+R[k]>>;
if (r<=mid) change(k<<,l,r,x,y,a);
else if (l>mid) change(k<<|,l,r,x,y,a);
else change(k<<,l,mid,x,y,a),change(k<<|,mid+,r,x,y,a);
}
int getans(int x,int y)
{
int k=,s=;
while ()
{
s=calc(s,query(root[k],,n,y,));
if (L[k]==R[k]) break;
if (x<=(L[k]+R[k]>>)) k<<=;
else k=k<<|;
}
return s%P;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj4785.in","r",stdin);
freopen("bzoj4785.out","w",stdout);
const char LL[]="%I64d";
#else
const char LL[]="%lld";
#endif
n=read(),m=read();
inv[]=;
for (int i=;i<=n;i++) inv[i]=(P-1ll*(P/i)*inv[P%i]%P)%P;
build(,,n);
for (int i=;i<=m;i++)
{
int op=read(),l=read(),r=read();
if (op==)
{
tot^=;int x=inv[r-l+];
change(,,l-,l,r,x);
if (r<n) change(,l,r,r+,n,x);
change(,l,r,l,r,x*%P);
}
else
{
if (l==) printf("%d\n",tot?(P+-getans(l-,r))%P:getans(l-,r));
else printf("%d\n",getans(l-,r));
}
}
return ;
}

BZOJ4785 ZJOI2017树状数组(概率+二维线段树)的更多相关文章

  1. [BZOJ4785][ZJOI2017]树状数组(概率+二维线段树)

    4785: [Zjoi2017]树状数组 Time Limit: 40 Sec  Memory Limit: 512 MBSubmit: 297  Solved: 195[Submit][Status ...

  2. 「ZJOI2017」树状数组(二维线段树)

    「ZJOI2017」树状数组(二维线段树) 吉老师的题目真是难想... 代码中求的是 \(\sum_{i=l-1}^{r-1}a_i\),而实际求的是 \(\sum_{i=l}^{r}a_i\),所以 ...

  3. BZOJ4785 [Zjoi2017]树状数组 【二维线段树 + 标记永久化】

    题目链接 BZOJ4785 题解 肝了一个下午QAQ没写过二维线段树还是很难受 首先题目中的树状数组实际维护的是后缀和,这一点凭分析或经验或手模观察可以得出 在\(\mod 2\)意义下,我们实际求出 ...

  4. P3688 [ZJOI2017] 树状数组 【二维线段树】

    题目描述:这里有一个写挂的树状数组: 有两种共\(m\)个操作: 输入\(l,r\),在\([l,r]\)中随机选择一个整数\(x\)执行\(\text{Add}(x)\) 输入\(l,r\),询问执 ...

  5. 洛谷 P3688 - [ZJOI2017]树状数组(二维线段树+标记永久化)

    题面传送门 首先学过树状数组的应该都知道,将树状数组方向写反等价于前缀和 \(\to\) 后缀和,因此题目中伪代码的区间求和实质上是 \(sum[l-1...n]-sum[r...n]=sum[l-1 ...

  6. luogu3380/bzoj3196 二逼平衡树 (树状数组套权值线段树)

    带修改区间K大值 这题有很多做法,我的做法是树状数组套权值线段树,修改查询的时候都是按着树状数组的规则找出那log(n)个线段树根,然后一起往下做 时空都是$O(nlog^2n)$的(如果离散化了的话 ...

  7. CF1093E Intersection of Permutations 树状数组套权值线段树

    \(\color{#0066ff}{ 题目描述 }\) 给定整数 \(n\) 和两个 \(1,\dots,n\) 的排列 \(a,b\). \(m\) 个操作,操作有两种: \(1\ l_a\ r_a ...

  8. BZOJ2141排队——树状数组套权值线段树(带修改的主席树)

    题目描述 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家 乐和和.红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别 ...

  9. Dynamic Rankings(树状数组套权值线段树)

    Dynamic Rankings(树状数组套权值线段树) 给定一个含有n个数的序列a[1],a[2],a[3]--a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i],a[i+1],a[ ...

  10. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

随机推荐

  1. Luogu P1962 斐波那契数列(矩阵乘法模板)

    传送门(其实就是求斐波那契数列....) 累了 明天再解释 做这道题需要一些关于矩阵乘法的基础知识. 1. 矩阵乘法的基础运算 只有当矩阵A的列数等于矩阵B的行数时,A与B可以相乘(A的行数不一定等于 ...

  2. Centos7下完美安装并配置mysql5.6

    Centos7将默认数据库mysql替换成了Mariadb,对于我们这些还想用mysql的人来说并不是一个好消息. 最近我搜罗了网上各种安装教程,各种出问题,要么安装失败,要么安装成功了却使用不了my ...

  3. Ubuntu 安装google chrome

    sudo apt-get install google-chrome-stable /usr/bin/google-chrome-stable

  4. [译]Kubernetes 分布式应用部署和人脸识别 app 实例

    原文地址:KUBERNETES DISTRIBUTED APPLICATION DEPLOYMENT WITH SAMPLE FACE RECOGNITION APP 原文作者:skarlso 译文出 ...

  5. ansible一键部署k8s单机环境

    一.虚拟机准备 干净的Centsot7.4.4G内存.2个CPU 最小化安装,最好带虚拟化 二.执行初始化脚本 注意:脚本中配置静态网卡根据实际网卡名称配置,我用的是ens33 可以用 sed -i ...

  6. 在Mac终端显示 Git 当前所在分支

    1.进入你的home目录 cd ~ 2.编辑.bashrc文件 vi .bashrc 3.将下面的代码加入到文件的最后处 function git_branch { branch="`git ...

  7. C. Make It Equal

    链接 [http://codeforces.com/contest/1065/problem/C] 题意 给你n个高度hi的塔,让你把高的部分切掉,使得最后所有塔一样高,而且每次切的高度之和不大于k ...

  8. A. Vasya and Chocolate

    链接 [http://codeforces.com/contest/1065/problem/A] 分析 一个公式完事 代码 #include<bits/stdc++.h> using n ...

  9. M1/M2阶段总结

    之前提问的博客 问题解答 问题 1 关于代码复审,复审者是否应该参与编码?如果复审者也参与编码的话,那么难免任务量较多,但如果不参与编码的话,工作分配的似乎不太均衡. 我们的团队项目在M1和M2阶段没 ...

  10. mybaits拦截器+自定义注解

    实现目的:为了存储了公共字典表主键的其他表在查询的时候不用关联查询(所以拦截位置位于mybaits语句查询得出结果集后) 项目环境 :springboot+mybaits 实现步骤:自定义注解——自定 ...