题目背景

在人类智慧的山巅,有着一台字长为1048576位(此数字与解题无关)的超级计算机,著名理论计算机科

学家P博士正用它进行各种研究。不幸的是,这天台风切断了电力系统,超级计算机

无法工作,而 P 博士明天就要交实验结果了,只好求助于学过OI的你. . . . . .

题目描述

P 博士将他的计算任务抽象为对一个整数的操作。

具体来说,有一个整数xx,一开始为00。

接下来有nn个操作,每个操作都是以下两种类型中的一种:

  • 1 a b:将x加上整数\(a\cdot 2^b\),其中a为一个整数,b为一个非负整数
  • 2 k :询问x在用二进制表示时,位权为2^k的位的值(即这一位上的1代表 2^k)

保证在任何时候,\(x\geqslant 0\)。

输入格式

输入的第一行包含四个正整数n,t_1,t_2,t_3,n的含义见题目描述,t_1,t_2,t_3的具体含义见子任务。

接下来n行,每行给出一个操作,具体格式和含义见题目描述。

同一行输入的相邻两个元素之间,用恰好一个空格隔开。

输出格式

对于每个询问操作,输出一行,表示该询问的答案(00或11)。对于加法操作,没有任何输出。

输入样例

10 3 1 2
1 100 0
1 2333 0
1 -233 0
2 5
2 7
2 15
1 5 15
2 15
1 -1 12
2 15

输出样例

0
1
0
1
0

Solution

奇怪的题...

注意到一个性质,二进制加法每次加一的均摊复杂度是\(O(1)\)的而不是\(O(\log n)\)的,具体可以考虑每一位被加了多少次。

那么其实我们可以压位然后暴力修改,但是均摊复杂度有一个缺点,就是不支持撤回操作,否则你可以在一次复杂度较高的操作前后跳转,然后时间复杂度就不对了。

这题的减法操作就可以视为是撤回操作。

那么我们可以维护两个高精度的数,分别表示一共加了多少减了多少,那么仔细思考一下就可以知道,对于询问我们唯一的瓶颈就是如何比较大小。

注意到字符串比较大小的暴力算法,我们只需要找到第一个不同的位就好了,那么我们可以开\(set\)来维护不同的位置,每次修改的时候更新一下就好了。

那么这题就变成了小清新模拟题,然后我调了一晚上

细节是真的挺多的....不过跑的还挺快...\(bzoj\) \(rk6\)。

#include<bits/stdc++.h>
using namespace std; void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} #define ui unsigned int const int maxn = 1e6+10; int n,_,s[maxn];
ui r[maxn][2]; set<int > dif; int cnt; int main() {
read(n),read(_),read(_),read(_);
for(int i=1;i<=n;i++) {
int op,a,b;read(op),read(a);
if(op==1) {
int k=0;read(b);
if(a<0) a=-a,k=1;
int t=b/32;b&=31;
ui p=(ui)a<<b,q=a>>(31-b),pre;q>>=1;
pre=r[t][k],r[t][k]+=p,q+=r[t][k]<pre;
if((r[t][k]^r[t][!k])&&!s[t]) s[t]=1,dif.insert(t);
else if(r[t][k]==r[t][!k]&&s[t]) s[t]=0,dif.erase(t);t++;
while(q) {
pre=r[t][k];r[t][k]+=q;q=r[t][k]<pre;
if((r[t][k]^r[t][!k])&&!s[t]) s[t]=1,dif.insert(t);
else if(r[t][k]==r[t][!k]&&s[t]) s[t]=0,dif.erase(t);
t++;
}
} else {
cnt++;
int t=a/32;a&=31;
int ans=((r[t][0]>>a)^(r[t][1]>>a))&1;
ui x=r[t][0]&((1u<<a)-1),y=r[t][1]&((1u<<a)-1);
if(x<y) write(ans^1);
else if(x>y||dif.empty()||t<=(*dif.begin())) write(ans);
else {
set<int > :: iterator it=dif.lower_bound(t);--it;
if(r[*it][0]>r[*it][1]) write(ans);
else write(ans^1);
}
}
}
return 0;
}

[BZOJ4942] [NOI2017]整数的更多相关文章

  1. [Bzoj4942][Noi2017]整数(线段树)

    4942: [Noi2017]整数 Time Limit: 50 Sec  Memory Limit: 512 MBSubmit: 363  Solved: 237[Submit][Status][D ...

  2. BZOJ4942 NOI2017整数(线段树)

    首先把每32位压成一个unsigned int(当然只要压起来能过就行).如果不考虑进/退位的话,每次只要将加/减上去的数拆成两部分直接单点修改就好了.那么考虑如何维护进/退位.可以发现进位的过程其实 ...

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

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

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

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

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

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

  6. 【BZOJ4942】[NOI2017]整数(分块)

    [BZOJ4942][NOI2017]整数(分块) 题面 BZOJ 洛谷 题解 暴力就是真正的暴力,直接手动模拟进位就好了. 此时复杂度是模拟的复杂度加上单次询问的\(O(1)\). 所以我们需要优化 ...

  7. [NOI2017]整数

    [NOI2017]整数 题目大意: \(n(n\le10^6)\)次操作维护一个长度为\(30n\)的二进制整数\(x\),支持以下两种操作: 将这个整数加上\(a\cdot2^b(|a|\le10^ ...

  8. NOI2017整数

    NOI2017 整数 题意: ​ 让你实现两个操作: 1 \(a\) \(b\):将\(x\)加上整数\(a \cdot 2 ^ b\),其中 \(a\)为一个整数,\(b\)为一个非负整数 2 \( ...

  9. 【bzoj4942】[Noi2017]整数 压位+线段树

    题目描述 P 博士将他的计算任务抽象为对一个整数的操作. 具体来说,有一个整数 $x$ ,一开始为0. 接下来有 $n$ 个操作,每个操作都是以下两种类型中的一种: 1 a b :将 $x$ 加上整数 ...

随机推荐

  1. VIO 初始化小结 - 10.17

    最近几个月忙于博士毕业,找工作一直没有继续更新博客,希望以这一篇开始,每个月能够继续有几篇总结博客. 首先review一下比较著名的vio系统 Tightly coupled EKF: mainly ...

  2. 限时购校验小工具&dubbo异步调用实现限

    本文来自网易云社区 作者:张伟 背景 限时购是网易考拉目前比较常用的促销形式,但是前期创建一个限时购活动时需要各个BU按照指定的Excel格式进行选品提报,为了保证提报数据准确,运营需要人肉校验很多信 ...

  3. 使用nmon监控得出网络实时速度以及最大、最小、平均网络传送速度

    首先我们得搞清楚几个概念,即什么是网速?什么是带宽? 举两个个例子: 1.家里装网线,宽带提供商说我们的带宽是100兆. 2.用迅雷下载电影,迅雷显示实时的下载速度是每秒3兆,或者说是3MB/s. 这 ...

  4. hdu1455Sticks(经典dfs+剪枝)

    Sticks Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  5. 基础的表ADT -数据结构(C语言实现)

    读数据结构与算法分析 表的概述 形如A1,A2,A3... 操作合集 PrintList MakeEmpty Find Insert Delete 表的简单数组实现 分析: PrintList和Fin ...

  6. 【第四章】Shell 条件测试表达式

    shell中条件测试的三种格式: 格式1: test 条件表达式格式2: [ 条件表达式 ]格式3: [[ 条件表达式 ]] 使用test: [root@host- ~]# test -f file ...

  7. linux服务器操作小技巧

    python程序后台一直运行,并将打印信息输出到文件中 nohup -u test.py > out.txt & -u 表示无缓冲,直接将打印信息输出带文件中 &表示程序后台运行

  8. Centos配置深度学习开发环境

    目录 1. 安装显卡驱动 2. 安装CUDA\CUDNN 3. 安装TensorFlow-gpu 测试 1. 安装显卡驱动 检测显卡驱动及型号 $ sudo rpm --import https:// ...

  9. LeetCode - 167. Two Sum II - Input array is sorted - O(n) - ( C++ ) - 解题报告

    1.题目大意 Given an array of integers that is already sorted in ascending order, find two numbers such t ...

  10. Python3 小工具-僵尸扫描

    僵尸机的条件: 1.足够闲置,不与其他机器进行通讯 2.IPID必须是递增的,不然无法判断端口是否开放 本实验僵尸机选择的是Windows2003 from scapy.all import * im ...