题目链接

UOJ 134

题解

可爱的电音之王松松松出的题……好妙啊。

首先想一个朴素的做法!

把当前的整数的二进制当作01序列用线段树维护一下(序列的第i位就是整数中位权为\(2^k\)的那一位)。

如何做加法?一下子加一个整数比较麻烦,可以把整数拆成一个个二进制位,一位位地加1。如果当前要加一的位置就是0,直接加就好了;否则显然要进位,松松松出的题肯定肯定不能暴力进位骗分(=v=)……所以线段树维护区间是否全是1,每次加的时候找右边(即更高位)第一个为0的位置,然后把那个位置修改为1,b和那个位置中间所有的位置都改成0就好了。

像这样(为了看着不反人类,最低位在右边,最高位在左边):

001111111
10
---------
010000001

减法怎么做呢?如果这一位就是1则直接减,否则找右边第一个为1的位置,然后单点修改为0,区间修改为1即可。

110000001
10
---------
101111111

但是这样做还是会T的!

于是要压位!

把30(或60?)个位用一个数存起来,然后类似上面的这样做。注意这时候不要还把a一位位拆开加了,最多只用拆成两部分(为了契合线段树中压位后每一“位”的大小)然后分别加就好了……

代码:

#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <queue>
#define space putchar(' ')
#define enter putchar('\n')
using namespace std;
typedef long long ll;
template <class T>
void read(T &x){
char c;
bool op = 0;
while(c = getchar(), c < '0' || c > '9')
if(c == '-') op = 1;
x = c - '0';
while(c = getchar(), c >= '0' && c <= '9')
x = x * 10 + c - '0';
if(op) x = -x;
}
template <class T>
void write(T x){
if(x < 0) putchar('-'), x = -x;
if(x >= 10) write(x / 10);
putchar(x % 10 + '0');
} const int N = 500005, S = 60;
const ll INF = (1LL << S) - 1;
int n, m, t, pos[4*N];
ll data[N], tag[4*N];
bool all[4*N][2]; void single_change(int k, ll x){
if(pos[k] != -1) data[pos[k]] = x;
if(x == 0) all[k][0] = 1, all[k][1] = 0, tag[k] = 0;
else if(x == INF) all[k][0] = 0, all[k][1] = 1, tag[k] = INF;
else all[k][0] = all[k][1] = 0, tag[k] = -1;
}
void pushdown(int k){
if(tag[k] == -1) return;
single_change(k << 1, tag[k]);
single_change(k << 1 | 1, tag[k]);
tag[k] = -1;
}
void pushup(int k){
all[k][0] = all[k << 1][0] & all[k << 1 | 1][0];
all[k][1] = all[k << 1][1] & all[k << 1 | 1][1];
}
void build(int k, int l, int r){
all[k][0] = 1, tag[k] = pos[k] = -1;
if(l == r) return (void)(pos[k] = l);
int mid = (l + r) >> 1;
build(k << 1, l, mid);
build(k << 1 | 1, mid + 1, r);
}
void range_change(int k, int l, int r, int ql, int qr, ll x){
if(ql <= l && qr >= r) return single_change(k, x);
pushdown(k);
int mid = (l + r) >> 1;
if(ql <= mid) range_change(k << 1, l, mid, ql, qr, x);
if(qr > mid) range_change(k << 1 | 1, mid + 1, r, ql, qr, x);
pushup(k);
}
int find_nxt(int k, int l, int r, int p, int o){
if(all[k][!o]) return -1;
if(l == r) return l;
pushdown(k);
int mid = (l + r) >> 1, tmp;
if(p <= mid && (tmp = find_nxt(k << 1, l, mid, p, o)) != -1) return tmp;
return find_nxt(k << 1 | 1, mid + 1, r, p, o);
}
ll query(int k, int l, int r, int p){
if(l == r) return data[l];
pushdown(k);
int mid = (l + r) >> 1;
if(p <= mid) return query(k << 1, l, mid, p);
else return query(k << 1 | 1, mid + 1, r, p);
}
void add(int p, ll x){
ll tmp = query(1, 0, n, p);
range_change(1, 0, n, p, p, (tmp + x) & INF);
if(tmp + x > INF){
int tar = find_nxt(1, 0, n, p + 1, 0);
range_change(1, 0, n, tar, tar, data[tar] + 1);
if(p + 1 <= tar - 1) range_change(1, 0, n, p + 1, tar - 1, 0);
}
}
void sub(int p, ll x){
ll tmp = query(1, 0, n, p);
range_change(1, 0, n, p, p, (tmp - x) & INF);
if(tmp - x < 0){
int tar = find_nxt(1, 0, n, p + 1, 1);
range_change(1, 0, n, tar, tar, data[tar] - 1);
if(p + 1 <= tar - 1) range_change(1, 0, n, p + 1, tar - 1, INF);
}
} int main(){ read(n), m = n, n = n / 2 + 2;
read(t), read(t), read(t);
build(1, 0, n);
ll op, a, b;
while(m--){
read(op), read(a);
if(op == 1){
read(b);
if(a > 0){
int p = b / S, rst = b % S;
ll x = a << rst & INF;
if(x) add(p, x);
p++, a >>= (S - rst);
if(b) add(p, a);
}
else{
a = -a;
int p = b / S, rst = b % S;
ll x = a << rst & INF;
if(x) sub(p, x);
p++, a >>= (S - rst);
if(b) sub(p, a);
}
}
else write(query(1, 0, n, a / S) >> (a % S) & 1), enter;
} return 0;
}

UOJ #314. 【NOI2017】整数 | 线段树 压位的更多相关文章

  1. [BZOJ4942][Noi2017]整数 线段树+压位

    用线段树来模拟加减法过程,维护连续一段中是否全为0/1. 因为数字很大,我们60位压一位来处理. #include<iostream> #include<cstring> #i ...

  2. 【BZOJ4942】[Noi2017]整数 线段树+DFS(卡过)

    [BZOJ4942][Noi2017]整数 题目描述去uoj 题解:如果只有加法,那么直接暴力即可...(因为1的数量最多nlogn个) 先考虑加法,比较显然的做法就是将A二进制分解成log位,然后依 ...

  3. 2018.10.30 bzoj4942: [Noi2017]整数(线段树压位)

    传送门 直接把修改的数拆成logloglog个二进制位一个一个修改是会TLETLETLE的. 因此我们把303030个二进制位压成一位储存在线段树里面. 然后维护区间中最靠左二进制位不为0/1的下标. ...

  4. 【洛谷3822】[NOI2017] 整数(线段树压位)

    题目: 洛谷 3822 分析: 直接按题意模拟,完了. 将每次加 / 减拆成不超过 \(32\) 个对单独一位的加 / 减. 考虑给一个二进制位(下称「当前位」)加 \(1\) 时,如果这一位本来就是 ...

  5. noi2017 T1 整数 ——线段树

    loj.ac上有  题目传送门 不过我还是把题目搬过来吧 整数(integer)[题目背景]在人类智慧的山巅,有着一台字长为 1048576 位的超级计算机,著名理论计算机科 学家 P 博士正用它进行 ...

  6. 【noi2017】 整数 线段树or模拟

    ORZYYB 题目大意:你需要维护一个有$3\times 10^7$个二进制位的数,有一种修改方式和一种询问方式 对这个数加上$a\times2^b$,其中$|a|≤10^9$,$b≤3\times ...

  7. BZOJ4946[Noi2017]蔬菜——线段树+堆+模拟费用流

    题目链接: [Noi2017]蔬菜 题目大意:有$n$种蔬菜,每种蔬菜有$c_{i}$个,每种蔬菜每天有$x_{i}$个单位会坏掉(准确来说每天每种蔬菜坏掉的量是$x_{i}-$当天这种蔬菜卖出量), ...

  8. POJ-2777-CountColor(线段树,位运算)

    链接:https://vjudge.net/problem/POJ-2777#author=0 题意: Chosen Problem Solving and Program design as an ...

  9. UOJ 217 奇怪的线段树

    http://uoj.ac/problem/217 题意就不X了,思路在这: 居然一开始把sap里面的mn设置为inf了,我是傻逼.. #include<cstdio> #include& ...

随机推荐

  1. WPF中DataGrid中的DataGridCheckBoxColumn用法(全选,全否,反选)

    原文:WPF中DataGrid中的DataGridCheckBoxColumn用法(全选,全否,反选) 前台代码 <DataGrid.Columns> <DataGridCheckB ...

  2. 多线程-volatile关键字和ThreadLocal

    1.并发编程中的三个概念 原子性:一个或多个操作.要么全部执行完成并且执行过程不会被打断,要么不执行.最常见的例子:i++/i--操作.不是原子性操作,如果不做好同步性就容易造成线程安全问题. 可见性 ...

  3. 使用Python遇到:'utf-8' codec can't decode byte 0x8b in position 1: invalid start byte 问题

    查看你的HTTP头部是否有如下头部信息:"Accept-Encoding": "gzip, deflate" 这条信息代表本地可以接收压缩格式的数据,而服务器在 ...

  4. JVM调优-GC参数

    一.Throughput收集器(吞吐量)-XX:+UseParallelGC-XX:+UseParallelOldGC *参数调整:通过调整堆大小,减少GC停顿时间,增大吞吐量增强堆大小可以减少Ful ...

  5. 浅谈博弈论中的两个基本模型——Bash Game&&Nim Game

    最近在数学这一块搞了蛮多题目,已经解决了数论基础,线性代数(只有矩阵,行列式待坑),组合数学中的一些简单问题.所以接下来不可避免的对博弈论这一哲学大坑开工. 当然,由于我很菜,所以也只能从最基础最容易 ...

  6. JVM规范系列第5章:加载、链接与初始化

    加载是根据特定名称查找类或接口类型的二进制表示(Binary Representation),并由此二进制表示创建类或接口的过程. 加载,就是指去寻找类或接口的过程. 链接是为了让类或接口可以被 Ja ...

  7. 常用rsync命令操作梳理

    作为一个运维工程师,经常可能会面对几十台.几百台甚至上千台服务器,除了批量操作外,环境同步.数据同步也是必不可少的技能.说到“同步”,不得不提的利器就是rsync.rsync不但可以在本机进行文件同步 ...

  8. beta阶段测试基本概况报告

    文件地址 测试基本信息                                                                                Bitmap 测试 ...

  9. Bootstrap Validator使用特性,动态(Dynamic)添加的input的验证问题

    http://1000hz.github.io/bootstrap-validator/#validator-usage Validated fields By default, the valida ...

  10. vue 路由传参 params 与 query两种方式的区别

    初学vue的时候,不知道如何在方法中跳转界面并传参,百度过后,了解到两种方式,params 与 query.然后,错误就这么来了:  router文件下index.js里面,是这么定义路由的: { p ...