nowcoder 211E - 位运算?位运算! - [二进制线段树][与或线段树]
题目链接:https://www.nowcoder.com/acm/contest/211/E
题目描述
请实现一个数据结构支持以下操作:区间循环左右移,区间与,区间或,区间求和。
输入描述:
第一行n,q表示数列长度及操作次数。
第二行n个数表示初始序列。
接下来q行表示操作。
操作格式如下:
一行表示一个操作。所有操作形如 opt  l  r  v。
opt=1 表示将区间[l,r]循环右移v位。
opt=2 表示将区间[l,r]循环左移v位。
opt=3 表示将区间[l,r]按位或上v。
opt=4 表示将区间[l,r]按位与上v。
opt=5 询问区间[l,r]的和。
保证opt=1或2时 1 ≤ v ≤ 20
注意:为了优化你的做题体验,操作5也会输入一个v,但是是没有意义的。
注意:循环左右移在20个二进制位的意义下进行
输出描述:
对于每个opt=5的操作,输出一个数表示答案。
输入
10 10
10112 23536 1305 7072 12730 29518 12315 3459 12435 29055
4 5 10 12373
2 1 6 7
5 4 10 24895
1 1 4 8
5 3 7 7767
5 7 9 6127
4 2 8 30971
5 4 10 2663
1 7 10 1
1 2 9 5
输出
2001530
1600111
24611
49482
备注:
1 ≤ N,Q ≤ 2*10^5 且 0 ≤ ai < 2^20
一些说明:
1. 对于00000000000000000101,右移一位后会变成10000000000000000010
2. 不是区间位移,是区间中的每一个数的二进制位的位移
题解:
(说实话这道题其实不难的,会线段树就应该想到怎么做的,可能是我脑子太笨了吧,比赛的时候想不到怎么做。)
考虑到只有二进制下只有 $20$ 位,而且修改操作都是按位与或者按位或,所以可以将 $20$ 位拆开来看。
然后就简单了呀:
对 $01$ 序列中某一个区间,全部与上 $x(x=0,1)$,那么 $x = 1$ 时不变, $x = 0$ 时全 $0$。
对 $01$ 序列中某一个区间,全部或上 $x(x=0,1)$,那么 $x = 0$ 时不变, $x = 1$ 时全 $1$。(这两种操作可以归结为直接把一个常见的线段树区间覆盖操作)
循环平移就是在 $20$ 棵平衡树进行数据的循环平移,向左向右循环平移都可以全部归结为向右平移,范围为 $(0,19)$,这个就是把每个线段树节点里的二十位的01序列重新排一下即可。
求和,我们线段树每个节点都有一个 $val[20]$ 用以为维护某一位上,当前区间内所有数字在这一位上合起来总共出现了多少个 $1$,也就是说这个 $val[20]$ 相当于一个算加法但还没来得及进位的二进制数,这样的二进制数转成十进制和普通二进制数转十进制是一样的。
AC代码:
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=2e5+; int n,q,a[maxn]; /********************************* Segment Tree - st *********************************/
int tmp[];
struct Node{
int l,r;
int val[];
int lazy[],shft;
void upd_shft(int x)
{
x=(x+)%;
shft=(shft+x)%;
for(int i=;i<x;i++) tmp[i]=lazy[i];
for(int i=;i<-x;i++) lazy[i]=lazy[i+x];
for(int i=;i<x;i++) lazy[i+-x]=tmp[i]; for(int i=;i<x;i++) tmp[i]=val[i];
for(int i=;i<-x;i++) val[i]=val[i+x];
for(int i=;i<x;i++) val[i+-x]=tmp[i];
}
void upd_lazy(int i,int x)
{
val[i]=(r-l+)*x;
lazy[i]=x;
}
}node[*maxn];
void pushdown(int root)
{
if(node[root].shft)
{
node[root*].upd_shft(node[root].shft);
node[root*+].upd_shft(node[root].shft);
node[root].shft=;
}
for(int i=;i<;i++)
{
if(node[root].lazy[i]!=-)
{
node[root*].upd_lazy(i,node[root].lazy[i]);
node[root*+].upd_lazy(i,node[root].lazy[i]);
node[root].lazy[i]=-;
}
}
}
void pushup(int root)
{
for(int i=;i<;i++)
node[root].val[i]=node[root*].val[i]+node[root*+].val[i];
}
void build(int root,int l,int r)
{
if(l>r) return;
node[root].l=l, node[root].r=r;
node[root].shft=;
for(int i=;i<;i++) node[root].lazy[i]=-;
if(l==r)
{
for(int i=;i<;i++)
node[root].val[i]=(a[l]>>i)%;
}
else
{
int mid=(l+r)/;
build(root*,l,mid);
build(root*+,mid+,r);
pushup(root);
}
}
void update(int type,int root,int st,int ed,int x)
{
if(st<=node[root].l && node[root].r<=ed)
{
if(type==)
{
node[root].upd_shft(x);
}
if(type==)
{
for(int i=,k;i<;i++)
{
k=(x>>i)%;
if(!k) node[root].upd_lazy(i,);
}
}
if(type==)
{
for(int i=,k;i<;i++)
{
k=(x>>i)%;
if(k) node[root].upd_lazy(i,);
}
}
}
else
{
int mid=(node[root].l+node[root].r)/;
pushdown(root);
if(st<=mid) update(type,root*,st,ed,x);
if(ed>mid) update(type,root*+,st,ed,x);
pushup(root);
}
}
ll query(int root,int st,int ed)
{
if(st<=node[root].l && node[root].r<=ed)
{
ll res=;
for(int i=;i<;i++) res+=(ll)node[root].val[i]*(<<i);
return res;
}
pushdown(root);
int mid=(node[root].l+node[root].r)/;
ll res=;
if(st<=mid) res+=query(root*,st,ed);
if(ed>mid) res+=query(root*+,st,ed);
return res;
}
/********************************* Segment Tree - ed *********************************/ int main()
{
scanf("%d%d",&n,&q);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
build(,,n);
while(q--)
{
int op,l,r,v;
scanf("%d%d%d%d",&op,&l,&r,&v);
if(op==) update(,,l,r,v);
if(op==) update(,,l,r,-v);
if(op==) update(,,l,r,v);
if(op==) update(,,l,r,v);
if(op==) printf("%lld\n",query(,l,r));
}
}
nowcoder 211E - 位运算?位运算! - [二进制线段树][与或线段树]的更多相关文章
- 算法竞赛进阶指南--lowbit运算,找到二进制下所有是1的位
		
// lowbit运算,找到二进制下所有是1的位 int H[37]; // 预处理 for (int i = 0; i < 36; i++) H[(1ll << i) % 37] ...
 - 位运算&字节运算
 - C#中的位的或运算的理解
		
如果懂位的运算,看到下面这2个程序执行的结果,会很容易理解,如果像我这样的菜鸟,刚接触开始肯定也觉得晕晕的,|= 这是什么运算符? |=就是位的或运算符,下面还是用上面的程序来讲解一下,为什么上面2个 ...
 - C166 8位字节位运算赋值-代码优化
		
8位字节位运算赋值优化特记录下: unsigned short func1(){ unsigned short a; return a;} unsigned char func2(){ unsigne ...
 - C++:用字符串数组实现大数运算,以两个不高于40位的大数运算为例。
		
因为基本数据类型中整型的内存范围有限,所以直接进行大数之间的运算,不仅浪费空间,而且运行缓慢,甚至有些会导致数据溢出. 那怎么办呢? 这时我们就想直接不行,那咱们来间接的. 这就是我们今天主要要讲的: ...
 - Python读字节某一位的值,设置某一位的值,二进制位操作
		
Python读字节某一位的值,设置某一位的值,二进制位操作 在物联网实际应用项目开发中,为了提升性能,与设备端配合,往往最终使用的是二进制字节串方式进行的通信协议封装,更会把0和1.True和Fa ...
 - JAVA如何把一个float四舍五入到小数点后2位,4位,或者其它指定位数.
		
怎么使float保留两位小数或多位小数 http://meryvn.blog.163.com/blog/static/36962664201173010402629/ 两种方法: import j ...
 - 中国石油大学(华东)暑期集训--二进制(BZOJ5294)【线段树】
		
问题 C: 二进制 时间限制: 1 Sec 内存限制: 128 MB提交: 8 解决: 2[提交] [状态] [讨论版] [命题人:] 题目描述 pupil发现对于一个十进制数,无论怎么将其的数字 ...
 - SQLite在.NET中自适应32位/64位系统
		
如果一个.NET应用要自适应32位/64位系统,只需要在项目的“目标平台”设置为“Any CPU”.但是如果应用中使用了SQLite,情况就不同了. SQLite的.NET开发包来自是System.D ...
 - 让使用SQLite的.NET应用自适应32位/64位系统
		
如果一个.NET应用要自适应32位/64位系统,只需要在项目的“目标平台”设置为“Any CPU”.但是如果应用中使用了SQLite,情况就不同了. SQLite的.NET开发包来自是System.D ...
 
随机推荐
- Linux下的两种磁盘分区工具的使用
			
如何使用fdisk和parted分区工具来进行硬盘分区,下面我来说一下在Linux系统中这两种硬盘分区工具的使用方法: ----------fdisk分区工具---------- ...
 - CMD递归文件夹
			
SET dir=%~dp0 SET /a cnt=0 echo dir is: %dir% cd /d %dir% for /R %dir% %%i in (*.apk) do ( set /a cn ...
 - Swift 类型桥接
			
前言 iOS 中的 API 基本都是在许多年前由 OC 写成的,现在通过桥接的方法在 Swift 中可以用,基本看不出区别,非常自然.但是一些特殊的类型,在两种语言进行桥接的时候需要特别注意. 1.N ...
 - 湾区求职分享:三个月刷题拿到 Google offer,欢迎踊跃提问
			
本文仅以个人经历和个人观点作为参考.如能受益,不胜荣幸. 本文会不断的修正,更新.希望通过大家的互动最后能写出一份阅者受益的文章. 本文纯手打,会有错别字,欢迎指出,虚心接受及时更改. 小马过河,大牛 ...
 - 【转】redis 消息队列发布订阅模式spring boot实现
			
最近做项目的时候写到一个事件推送的场景.之前的实现方式是起job一直查询数据库,看看有没有最新的消息.这种方式非常的不优雅,反正我是不能忍,由于羡慕本身就依赖redis,刚好redis 也有消息队列的 ...
 - [k8s]docker calico网络&docker cluster-store
			
docker cluster-store选项 etcd-calico(bgp)实现docker夸主机通信 配置calico网络 - 启动etcd etcd --listen-client-urls h ...
 - git合并多个提交
			
git合并多个提交 [时间:2016-11] [状态:Open] [关键词:git,git rebase,合并提交,commit] 0. 引言 本文是关于Git提交记录修改的方法,主要是将多个提交记录 ...
 - java InputStream和OutputStream
			
InputStream类型 类 功能 构造器参数 如何使用 ByteArrayInputStream 允许将内存的缓冲区当做InputStreams使用 缓冲区,字节将从中取出 作为一种数据源:将其与 ...
 - JavaScript高级用法一之事件响应与网页交互
			
综述 本篇的主要内容来自慕课网,事件响应与网页交互,主要内容如下 1 什么是事件 2 鼠标单击事件( onclick ) 3 鼠标经过事件(onmouseover) 4 鼠标移开事件(onmouseo ...
 - altium designer 10如何画4层板
			
本篇博客主要讲解一下如何用altium designer10去画4层板. 想想当初自己画4层板时,也去网上海找资料,结果是零零散散,也没讲出个123,于是硬着头皮去找师兄,如何画4层板.师兄冷笑道:“ ...