「ZJOI2017」树状数组

以下均基于模2意义下,默认\(n,m\)同阶。

熟悉树状数组的应该可以发现,这题其实是求\(l-1\)和\(r\)位置值相同的概率。

显然\(l=1\)的情况需要特盘。

大暴力

对于\(l=1\)的情况,可以发现一个操作不会产生影响当且仅当增加\(r\)的值,而其他情况会改变\(l-1\)或\(r\)。

对于\(l!=1\)的情况:

​ 针对一次修改区间\([ql,qr]\)。

  1. \([ql,qr]\)包含\(l-1,r\),那么有\(\displaystyle 2 \over qr-ql+1\)概率使\(l-1\)或\(r\)的值改变。
  2. \([ql,qr]\)不包含\(l-1,r\),不会发生变化。
  3. \([ql,qr]\)包含\(l-1,r\)中一个,那么有\(\displaystyle 1 \over qr-ql+1\)使\(l-1\)或\(r\)的值改变。

把每次修改记下来,就可以写出\(o(n^2)\)的暴力了。

代码见namespace fc​

正解

显然可以只记相等的概率。

可以发现,对于一次询问,改变前面修改的顺序并不会改变该询问的答案。

也就是说它满足交换律。

对于\(l=1\)的情况,显然可以一棵线段树维护。

对于\(l!=1\)的情况,把\([l,r ]\)区间当做一个二维的点\((l,r)\),那么每一次修改都会对二维区间产生贡献。

具体的:对于暴力分的第1类,即\(x \in [ql,qr],y \in [ql,qr]\)的点,对于暴力的第3类,即\(x \in [1,ql -1],y \in [ql,qr]\),\(x \in [ql,qr],y \in [qr+1,n]\)。

只需二维线段树区间标记即可,为了方便,标记表示的是 对应的二维区间的点 发生变化的概率。

#include<bits/stdc++.h>
#define rep(q,a,b) for(int q=a,q##_end_=b;q<=q##_end_;++q)
#define dep(q,a,b) for(int q=a,q##_end_=b;q>=q##_end_;--q)
#define mem(a,b) memset(a,b,sizeof a )
#define debug(a) cerr<<#a<<' '<<a<<"___"<<endl
using namespace std;
template<typename T>
void in(T &r) {
static char c;
r=0;
while(c=getchar(),!isdigit(c));
do r=(r<<1)+(r<<3)+(c^48);
while(c=getchar(),isdigit(c));
}
bool cur1;
int n,m;
const int mn=100005;
const int mod=998244353;
int inv[mn];
namespace fc{
int l[3005],r[3005];
void solve(){
int ty,a,b;
int ct=0;
while(m--){
in(ty),in(a),in(b);
if(ty==1)++ct,l[ct]=a,r[ct]=b;
else{
--a;
int wi=1,wo=0;
if(!a){
rep(q,1,ct){
int len=r[q]-l[q]+1;
if(l[q]<=b&&b<=r[q]){
int mid1=wi,mid2=wo;
wi=(1LL*mid2*(1-inv[len])+1LL*mid1*inv[len])%mod;
wo=(1LL*mid1*(1-inv[len])+1LL*mid2*inv[len])%mod;
}else swap(wi,wo);
}
}else{
rep(q,1,ct){
int len=r[q]-l[q]+1;
int mid1=wi,mid2=wo;
if(l[q]<=a&&b<=r[q]){//[l[q],r[q]]->[l[q],r[q]]
wi=(1LL*mid1*(1-2*inv[len])+1LL*mid2*2*inv[len])%mod;
wo=(1LL*mid2*(1-2*inv[len])+1LL*mid1*2*inv[len])%mod;
}else if(l[q]<=b&&b<=r[q]||l[q]<=a&&a<=r[q]){
//[1,l[q]-1]->[l[q],r[q]]
//[l[q],r[q]]->[r[q]+1,n]
wi=(1LL*mid1*(1-inv[len])+1LL*mid2*inv[len])%mod;
wo=(1LL*mid2*(1-inv[len])+1LL*mid1*inv[len])%mod;
}
}
}
printf("%d\n",(wi+mod)%mod);
}
}
}
}
namespace something_just_for_fun{
struct two_dimensional_segment_tree{
int tot,lson[mn*400],rson[mn*400],addv[mn*400],rt[mn<<2];
two_dimensional_segment_tree(){
tot=0;
}
int y_1,y_2,y_3,y_4,ad_v;
void se_add(int &o,int l,int r){
if(!o)o=++tot;
if(y_3<=l&&r<=y_4){
addv[o]=(1LL*(1-addv[o])*ad_v+1LL*addv[o]*(1-ad_v))%mod;
}else{
int mid=l+r>>1;
if(y_3<=mid)se_add(lson[o],l,mid);
if(y_4>mid)se_add(rson[o],mid+1,r);
}
}
void fi_add(int o,int l,int r){
if(y_1<=l&&r<=y_2)se_add(rt[o],1,n);
else{
int mid=l+r>>1;
if(y_1<=mid)fi_add(o<<1,l,mid);
if(y_2>mid)fi_add(o<<1|1,mid+1,r);
}
}
void add(int l,int r,int l1,int r1,int v){
if(l>r||l1>r1)return;
y_1=l,y_2=r,y_3=l1,y_4=r1,ad_v=v;
fi_add(1,1,n);
}
int v;
void se_ask(int &o,int l,int r){
if(!o)return;
v=(1LL*(1-addv[o])*v+1LL*addv[o]*(1-v))%mod;
int mid=l+r>>1;
if(y_2<=mid)se_ask(lson[o],l,mid);
else se_ask(rson[o],mid+1,r);
}
void fi_ask(int o,int l,int r){
se_ask(rt[o],1,n);
if(l==r)return;
else{
int mid=l+r>>1;
if(y_1<=mid)fi_ask(o<<1,l,mid);
else fi_ask(o<<1|1,mid+1,r);
}
}
int ask(int l,int r){
y_1=l,y_2=r,v=1;
fi_ask(1,1,n);
return v;
}
}an;
struct segment_tree{
int addv[mn<<2];
int y_1,y_2,ad_v;
void fi_add(int o,int l,int r){
if(y_1<=l&&r<=y_2)addv[o]=(1LL*(1-addv[o])*ad_v+1LL*addv[o]*(1-ad_v))%mod;
else{
int mid=l+r>>1;
if(y_1<=mid)fi_add(o<<1,l,mid);
if(y_2>mid)fi_add(o<<1|1,mid+1,r);
}
}
void add(int l,int r,int v){
if(l>r)return;
y_1=l,y_2=r,ad_v=v;
fi_add(1,1,n);
}
int v;
void fi_ask(int o,int l,int r){
v=(1LL*(1-addv[o])*v+1LL*addv[o]*(1-v))%mod;
if(l==r)return;
else{
int mid=l+r>>1;
if(y_1<=mid)fi_ask(o<<1,l,mid);
else fi_ask(o<<1|1,mid+1,r);
}
}
int ask(int x){
y_1=x,v=1;
fi_ask(1,1,n);
return v;
}
}at;
int l[mn],r[mn];
void solve(){
int ty,a,b;
while(m--){
in(ty),in(a),in(b);
if(ty==1){
at.add(a,b,1-inv[b-a+1]),at.add(1,a-1,1),at.add(b+1,n,1);
an.add(a,b,a,b,2*inv[b-a+1]%mod),an.add(1,a-1,a,b,inv[b-a+1]),an.add(a,b,b+1,n,inv[b-a+1]);
}
else --a,printf("%d\n",((!a?at.ask(b):an.ask(a,b))+mod)%mod);
}
}
}
bool cur2;
int main(){
// cerr<<(&cur2-&cur1)/1024.0/1024<<endl;
freopen("bit.in","r",stdin);
freopen("bit.out","w",stdout);
in(n),in(m);
inv[0]=inv[1]=1;
rep(q,2,n)inv[q]=1LL*(mod-mod/q)*inv[mod%q]%mod;
something_just_for_fun::solve();
return 0;
}

「ZJOI2017」树状数组的更多相关文章

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

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

  2. UOJ#291. 【ZJOI2017】树状数组 树套树

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ291.html 题解 结论:这个写错的树状数组支持的是后缀加和后缀求和.这里的后缀求和在 x = 0 的时 ...

  3. 【ZJOI2017】树状数组

    题目描述 漆黑的晚上,九条可怜躺在床上辗转反侧.难以入眠的她想起了若干年前她的一次悲惨的 OI 比赛经历.那是一道基础的树状数组题. 给出一个长度为 $n$ 的数组 $A$,初始值都为 $0$,接下来 ...

  4. loj #535. 「LibreOJ Round #6」花火 树状数组求逆序对+主席树二维数点+整体二分

    $ \color{#0066ff}{ 题目描述 }$ 「Hanabi, hanabi--」 一听说祭典上没有烟火,Karen 一脸沮丧. 「有的哦-- 虽然比不上大型烟花就是了.」 还好 Shinob ...

  5. 「CodePlus 2017 11 月赛」Yazid 的新生舞会(树状数组/线段树)

    学习了新姿势..(一直看不懂大爷的代码卡了好久T T 首先数字范围那么小可以考虑枚举众数来计算答案,设当前枚举到$x$,$s_i$为前$i$个数中$x$的出现次数,则满足$2*s_r-r > 2 ...

  6. 「CodePlus 2017 12 月赛」火锅盛宴(模拟+树状数组)

    1A,拿来练手的好题 用一个优先队列按煮熟时间从小到大排序,被煮熟了就弹出来. 用n个vector维护每种食物的煮熟时间,显然是有序的. 用树状数组维护每种煮熟食物的数量. 每次操作前把优先队列里煮熟 ...

  7. 【loj6041】「雅礼集训 2017 Day7」事情的相似度 后缀自动机+STL-set+启发式合并+离线+扫描线+树状数组

    题目描述 给你一个长度为 $n$ 的01串,$m$ 次询问,每次询问给出 $l$ .$r$ ,求从 $[l,r]$ 中选出两个不同的前缀的最长公共后缀长度的最大值. $n,m\le 10^5$ 题解 ...

  8. 「模拟赛20180306」回忆树 memory LCA+KMP+AC自动机+树状数组

    题目描述 回忆树是一棵树,树边上有小写字母. 一次回忆是这样的:你想起过往,触及心底--唔,不对,我们要说题目. 这题中我们认为回忆是这样的:给定 \(2\) 个点 \(u,v\) (\(u\) 可能 ...

  9. 「模拟赛20180307」三元组 exclaim 枚举+树状数组

    题目描述 给定 \(n,k\) ,求有多少个三元组 \((a,b,c)\) 满足 \(1≤a≤b≤c≤n\)且\(a + b^2 ≡ c^3\ (mod\ k)\). 输入 多组数据,第一行数据组数\ ...

随机推荐

  1. github项目托管方式(看项目自身是否自带有 .git)

    一.本地仓库和远程仓库建立联系 方式一:项目自身带有 .git文件的[自身就是一个本地仓库的](这里咱以vue-cli3项目为例) 1.创建自带.git本地仓库:创建一个叫 my-vue 的项目 2. ...

  2. Exponential family of distributions

    目录 定义 性质 极大似然估计 最大熵 例子 Bernoulli 指数分布 正态分布 Choi H. I. Lecture 4: Exponential family of distributions ...

  3. [opencv]调用鼠标事件执行grabcut算法实现阈值分割

    #include<iostream> #include <opencv2/opencv.hpp> #include <math.h> using namespace ...

  4. http协议的状态码400,401,403,404,500,502,503,301,302等常见网页错误代码

    1xx(临时响应)表示临时响应并需要请求者继续执行操作的状态码. 100(继续) 请求者应当继续提出请求.服务器返回此代码表示已收到请求的第一部分,正在等待其余部分. 101(切换协议) 请求者已要求 ...

  5. 「影院售票系统」 · Java Swing + MySQL JDBC开发

    目录 文档说明: 一.语言和环境 二.实现功能 三.数据库设计 四.具体要求及推荐实现步骤 五.注意事项 六.评分标准 实现代码: 一.数据库: 二.Java Swing: com.ynavc.Bea ...

  6. Java 私有接口 【类中嵌套接口】

    1.前言 接口十分常用,能规范实现类的命名 和 实现多个实现类的向上转型成统一类型 ,但是接口的修饰符只能是 public吗? 当然不是,可以是private , 难道是像这样? 显然不可以,已经报错 ...

  7. nuxt写路由接口

    //在server/interface/city.js import Router from 'koa-router'; const router = new Router({ prefix:'/ci ...

  8. Word2010格式化可爱的家乡

    原文链接:https://www.toutiao.com/i6487795632349118990/ 准备样文 选中"可爱的家乡",选择"开始"选项卡,&quo ...

  9. iview获取全选和半选数据(方法一:基于树型数据)

    在用iview的时候发现iview的树中获取半选和全选的函数getCheckedAndIndeterminateNodes在我使用的iview版本里面是没有提供的, 于是自己写了一下获取全选和半选节点 ...

  10. C# 开源一个基于 yarp 的 API 网关 Demo,支持绑定 Kubernetes Service

    关于 Neting 刚开始的时候是打算使用微软官方的 Yarp 库,实现一个 API 网关,后面发现坑比较多,弄起来比较麻烦,就放弃了.目前写完了查看 Kubernetes Service 信息.创建 ...