[bzoj5294]二进制
首先可以发现$2^k$模3意义下有循环节,也就是1,-1,1,-1……
考虑对于x个1,y个0,判断是否存在3的倍数
1.x为偶数时一定可以,选择等量的1和-1即可
2.x为奇数,要满足$x\ge 3$且$y\ge 2$,这是可以用3个0*(-1)和3个1*1来抵消掉(如果y=2时也可以,因为此时总共有奇数位),同时$x-3$显然为偶数
看上去难以维护,考虑反过来,求不是3的倍数的区间个数,那么即要求$x=1$或x为奇数且$y=0/1$
线段树维护区间,对于每一个区间维护0的数量,1的数量和一个三维数组a[3][4][3]表示在任意/以左端点为开头/右端点为结尾的区间中,x为0/1/非0偶数/非1奇数,y为0/1/(>1)的数量,转移详见代码(比较丑陋)
1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 100005
4 #define L (k<<1)
5 #define R (L+1)
6 #define mid (l+r>>1)
7 #define ll long long
8 struct ji{
9 int s[2];
10 ll a[3][4][3];
11 }o,f[N<<2];
12 int n,m,p,x,y,a[N];
13 int fx(int x){
14 if (x<2)return x;
15 return ((x&1)+2);
16 }
17 int fy(int y){
18 return min(y,2);
19 }
20 ji up(ji x,ji y){
21 if (x.s[0]<0)return y;
22 if (y.s[0]<0)return x;
23 for(int i=0;i<2;i++)o.s[i]=x.s[i]+y.s[i];
24 memset(o.a[0],0,sizeof(o.a[0]));
25 memcpy(o.a[1],x.a[1],sizeof(o.a[1]));
26 memcpy(o.a[2],y.a[2],sizeof(o.a[2]));
27 for(int i1=0;i1<4;i1++)
28 for(int j1=0;j1<3;j1++){
29 o.a[0][i1][j1]+=x.a[0][i1][j1]+y.a[0][i1][j1];
30 o.a[1][fx(i1+x.s[1])][fy(j1+x.s[0])]+=y.a[1][i1][j1];
31 o.a[2][fx(i1+y.s[1])][fy(j1+y.s[0])]+=x.a[2][i1][j1];
32 for(int i2=0;i2<4;i2++)
33 for(int j2=0;j2<3;j2++)
34 o.a[0][fx(i1+i2)][fy(j1+j2)]+=x.a[2][i1][j1]*y.a[1][i2][j2];
35 }
36 return o;
37 }
38 void update(int k,int l,int r,int x,int y){
39 if (l==r){
40 f[k].s[y]=1;
41 f[k].s[y^1]=0;
42 memset(f[k].a,0,sizeof(f[k].a));
43 for(int i=0;i<3;i++)f[k].a[i][y][y^1]=1;
44 return;
45 }
46 if (x<=mid)update(L,l,mid,x,y);
47 else update(R,mid+1,r,x,y);
48 f[k]=up(f[L],f[R]);
49 }
50 ji query(int k,int l,int r,int x,int y){
51 if ((l>y)||(x>r))return f[0];
52 if ((x<=l)&&(r<=y))return f[k];
53 return up(query(L,l,mid,x,y),query(R,mid+1,r,x,y));
54 }
55 int main(){
56 scanf("%d",&n);
57 for(int i=1;i<=n;i++)scanf("%d",&a[i]);
58 for(int i=1;i<=n;i++)update(1,1,n,i,a[i]);
59 f[0].s[0]=-1;
60 scanf("%d",&m);
61 for(int i=1;i<=m;i++){
62 scanf("%d%d",&p,&x);
63 if (p==1)update(1,1,n,x,a[x]^=1);
64 else{
65 scanf("%d",&y);
66 ll ans=(y-x+2LL)*(y-x+1)/2;
67 o=query(1,1,n,x,y);
68 for(int j=0;j<3;j++)ans-=o.a[0][1][j];
69 for(int j=0;j<2;j++)ans-=o.a[0][3][j];
70 printf("%lld\n",ans);
71 }
72 }
73 }
[bzoj5294]二进制的更多相关文章
- 【BZOJ5294】[BJOI2018]二进制(线段树)
[BZOJ5294][BJOI2018]二进制(线段树) 题面 BZOJ 洛谷 题解 二进制串在模\(3\)意义下,每一位代表的余数显然是\(121212\)这样子交替出现的. 其实换种方法看,就是\ ...
- 2019.02.12 bzoj5294: [Bjoi2018]二进制(线段树)
传送门 题意简述: 给出一个长度为nnn的二进制串. 你需要支持如下操作: 修改每个位置:1变0,0变1 询问对于一个区间的子二进制串有多少满足重排之后转回十进制值为333的倍数(允许前导000). ...
- BZOJ5294 BJOI2018二进制(线段树)
二进制数能被3整除相当于奇数.偶数位上1的个数模3同余.那么如果有偶数个1,一定存在重排方案使其合法:否则则要求至少有两个0且至少有3个1,这样可以给奇数位单独安排3个1. 考虑线段树维护区间内的一堆 ...
- 中国石油大学(华东)暑期集训--二进制(BZOJ5294)【线段树】
问题 C: 二进制 时间限制: 1 Sec 内存限制: 128 MB提交: 8 解决: 2[提交] [状态] [讨论版] [命题人:] 题目描述 pupil发现对于一个十进制数,无论怎么将其的数字 ...
- BZOJ5294 BJOI2018 二进制 线段树
传送门 因为每一位\(\mod 3\)的值为\(1,2,1,2,...\),也就相当于\(1,-1,1,-1,...\) 所以当某个区间的\(1\)的个数为偶数的时候,一定是可行的,只要把这若干个\( ...
- BZOJ5294 [BJOI2018] 二进制 【线段树】
BJOI的题目感觉有点难写 题目分析: 首先推一波结论.接下来的一切都在模3意义下 现在我们将二进制位重组,不难发现的是2^0≡1,2^1≡2,2^2≡1,2^3≡2....所以我们考虑这样的式子 2 ...
- Bzoj5294/洛谷P4428 [Bjoi2018]二进制(线段树)
题面 Bzoj 洛谷 题解 考虑一个什么样的区间满足重组之后可以变成\(3\)的倍数.不妨设\(tot\)为一个区间内\(1\)的个数.如果\(tot\)是个偶数,则这个区间一定是\(3\)的倍数,接 ...
- 使用struct处理二进制
有的时候需要用python处理二进制数据,比如,存取文件.socket操作时.这时候,可以使用python的struct模块来完成. struct模块中最重要的三个函数是pack(), unpack( ...
- 如何开启MySQL 5.7.12 的二进制日志
1. 打开/etc下的my.cnf文件 2. 编辑它,添加内容: log_bin=binary-log #二进制日志的文件名 server_id=1 #必须指定server_id,这是MySQL ...
随机推荐
- react之组建通信
父组件与子组件通信 父组件将自己的状态传递给子组件,子组件当做属性来接收,当父组件更改自己状态的时候,子组件接收到的属性就会发生改变 父组件利用ref对子组件做标记,通过调用子组件的方法以更改子组件的 ...
- yum源安装nginx
nginx使用yum源安装 安装步骤 使用yum源安装依赖 yum install yum-utils 配置nginx.repo的yum文件 vim /etc/yum.repos.d/nginx.re ...
- C++手动加载CLR运行托管程序(CLR Hosting)
转载自:http://www.linuxidc.com/Linux/2012-10/72293.htm 机制介绍 有些时候主程序是通过C/C++实现的,但是我们希望通过托管代码来扩展非托管程序,从而也 ...
- python的参数传递是值传递还是引用传递??
函数参数传递机制,传值和引用的是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(pa ...
- linux系统(centos)下su和sudo命令的区别
linux系统(centos)下su和sudo命令的区别 区别 我们在日常使用过程中,这2个命令很多时候能达到相同的效果,对细节区别十分模糊,这里进行简单的解释和区分.希望大家能够正确使用这2个命令, ...
- Google Object detection配置与使用
Google Object detection 前言: 本文记录了使用Google发布的Object detection(July 1st, 2019)接口,完成了对标注目标的检测.参考了很多博文,在 ...
- the Agiles Scrum Meeting 4
会议时间:2020.4.12 20:00 1.每个人的工作 今天已完成的工作 yjy:基本完成广播功能,修复bug issues:小组任务1-增量开发组 Bug:冲刺 wjx:继续实现注销功能的后端 ...
- lib库无法加载的情况分析
最近升级vs2017的时候遇到无法加载库的问题,在网上查找问题,网上给出可能有三种情况导致该问题:路径是否正确:库依赖是否齐全:库版本是否正确.最直接的方法就是用depends软件去查询,是否有模块有 ...
- allegro查看线宽的方法
- Hash算法:双重散列
双重散列是线性开型寻址散列(开放寻址法)中的冲突解决技术.双重散列使用在发生冲突时将第二个散列函数应用于键的想法. 此算法使用: (hash1(key) + i * hash2(key)) % TAB ...