标签:utc read define 概念 div query img out include
漆黑的晚上,九条可怜躺在床上辗转反侧。难以入眠的她想起了若干年前她的一次悲惨的 OI 比赛经历。那是一道基础的树状数组题。
给出一个长度为 nn 的数组 AA,初始值都为 00,接下来进行 mm 次操作,操作有两种:
尽管那个时候的可怜非常的 simple,但是她还是发现这题可以用树状数组做。当时非常 young 的她写了如下的算法:
其中 lowbit(x)lowbit(x) 表示数字 xx 最低的非 00 二进制位,例如 lowbit(5)=1,lowbit(12)=4lowbit(5)=1,lowbit(12)=4。进行第一类操作的时候就调用 Add(x)Add(x),第二类操作的时候答案就是 Query(l,r)Query(l,r)。
如果你对树状数组比较熟悉,不难发现可怜把树状数组写错了:AddAdd 和 FindFind 中 xx 变化的方向反了。因此这个程序在最终测试时华丽的爆 00 了。
然而奇怪的是,在当时,这个程序通过了出题人给出的大样例——这也是可怜没有进行对拍的原因。
现在,可怜想要算一下,这个程序回答对每一个询问的概率是多少,这样她就可以再次的感受到自己是一个多么非的人了。然而时间已经过去了很多年,即使是可怜也没有办法完全回忆起当时的大样例。幸运的是,她回忆起了大部分内容,唯一遗忘的是每一次第一类操作的 xx 的值,因此她假定这次操作的 xx 是在 [li,ri][li,ri] 范围内等概率随机的。
具体来说,可怜给出了一个长度为 nn 的数组 AA,初始为 00,接下来进行了 mm 次操作:
第一行输入两个整数 n,mn,m。 接下来 mm 行每行描述一个操作,格式如题目中所示。
对于每组询问,输出一个整数表示答案。如果答案化为最简分数后形如 xyxy,那么你只需要输出 x×y−1mod998244353x×y−1mod998244353 后的值。(即输出答案模 998244353998244353)。
1 #include<bits/stdc++.h> 2 #define rg register 3 #define ll long long 4 using namespace std; 5 const int N=100010,mod=998244353; 6 int n,m,sz1,sz2,Ls[N*400],Rs[N*400],lz[N*400],rt[N<<2],ls[N<<2],rs[N<<2],Rt,ans; 7 int inv(ll x){ 8 ll re=1; 9 for(int y=mod-2;y;y>>=1,x=x*x%mod){ 10 if(y&1)re=re*x%mod; 11 } 12 return re; 13 } 14 char gc(){ 15 static char*p1,*p2,s[1000000]; 16 if(p1==p2)p2=(p1=s)+fread(s,1,1000000,stdin); 17 return(p1==p2)?EOF:*p1++; 18 } 19 int rd(){ 20 int x=0; char c=gc(); 21 while(c<‘0‘||c>‘9‘)c=gc(); 22 while(c>=‘0‘&&c<=‘9‘)x=x*10+c-‘0‘,c=gc(); 23 return x; 24 } 25 void update(int&k,int l,int r,int x,int y,int p){ 26 if(!k)lz[k=++sz2]=1; 27 if(l==x&&r==y){ 28 lz[k]=((ll)lz[k]*(mod+1-p)+(ll)(mod+1-lz[k])*p)%mod; 29 return ; 30 } 31 else{ 32 int mid=(l+r)>>1; 33 if(y<=mid)update(Ls[k],l,mid,x,y,p); 34 else if(x>mid)update(Rs[k],mid+1,r,x,y,p); 35 else update(Ls[k],l,mid,x,mid,p),update(Rs[k],mid+1,r,mid+1,y,p); 36 } 37 } 38 void query(int k,int l,int r,int x){ 39 if(!k)return; 40 ans = ((ll)ans*lz[k] + (ll)(mod+1-ans)*(mod+1-lz[k]))%mod; 41 if(l==r)return; 42 else{ 43 int mid=(l+r)>>1; 44 if(x<=mid)query(Ls[k],l,mid,x); 45 else query(Rs[k],mid+1,r,x); 46 } 47 } 48 void Update(int&k,int l,int r,int x1,int x2,int y1,int y2,int p){ 49 if(!k)k=++sz1; 50 if(l==x1&&r==x2)update(rt[k],1,n,y1,y2,p); 51 else{ 52 int mid=(l+r)>>1; 53 if(x2<=mid)Update(ls[k],l,mid,x1,x2,y1,y2,p); 54 else if(x1>mid)Update(rs[k],mid+1,r,x1,x2,y1,y2,p); 55 else Update(ls[k],l,mid,x1,mid,y1,y2,p),Update(rs[k],mid+1,r,mid+1,x2,y1,y2,p); 56 } 57 } 58 void Query(int k,int l,int r,int x,int y){ 59 if(!k)return; 60 query(rt[k],1,n,y); 61 if(l==r)return; 62 else{ 63 int mid=(l+r)>>1; 64 if(x<=mid)Query(ls[k],l,mid,x,y); 65 else Query(rs[k],mid+1,r,x,y); 66 } 67 } 68 int cnt,a[N]; 69 inline void print(int x){ 70 if(!x){putchar(48);putchar(10);return;} 71 cnt=0; 72 while(x)a[++cnt]=x%10,x/=10; 73 for(rg int i=cnt;i;--i)putchar(a[i]+48); 74 putchar(10); 75 } 76 int main(){ 77 #ifndef ONLINE_JUDGE 78 freopen("T2.in","r",stdin); 79 freopen("T2.out","w",stdout); 80 #endif 81 n=rd();m=rd(); 82 for(rg int i=1,l,r,op;i<=m;++i){ 83 op=rd();l=rd();r=rd(); 84 if(op==1){ 85 int p = inv(r-l+1); 86 if(l>1)Update(Rt,1,n,1,l-1,l,r,p),update(rt[0],1,n,1,l-1,1); 87 if(r<n)Update(Rt,1,n,l,r,r+1,n,p),update(rt[0],1,n,r+1,n,1); 88 if(l<r)Update(Rt,1,n,l,r,l,r,p*2%mod),update(rt[0],1,n,l,r,(mod+1-p)%mod); 89 }else{ 90 ans=1; 91 if(l==1)query(rt[0],1,n,r); 92 else Query(Rt,1,n,l-1,r); 93 print(ans); 94 // printf("%d\n",ans); 95 } 96 } 97 // printf("%.2lf\n",1.0*clock()/CLOCKS_PER_SEC); 98 return 0; 99 }
标签:utc read define 概念 div query img out include
原文地址:https://www.cnblogs.com/Paul-Guderian/p/10307005.html