「NOI2017」整数

有一些比较简单的\(\log^2n\)做法

比如暴力在动态开点线段树上维护每个位置为\(0\)还是\(1\),我们发现涉及到某一位加上\(1\)或者减去\(1\)实际上对其他位的影响只有区间覆盖,通过线段树上二分可以得到区间覆盖的位置,然后暴力区间覆盖即可。

反正我这种菜鸡大常数写法只得到了68分..


考虑利用势能,注意到如果同时改变加法和减法,势能很容易被\(b\)搞掉

如果分开维护加法和减法,把位置上的\(1\)的个数当做势能,可以发现,暴力修改是均摊\(O(n\log a)\)的

直接暴力维护两个数组\(plu\)和\(dec\)

考虑单点求值,因为保证了\(x\ge 0\)

所以我们发现只需要两个条件就可以确定第\(k\)位的值,即

  • \(ans_1=[plu_k=dec_k]\)
  • \(ans_2=[plu[k-1,1]\ge[dec[k-1,1]]]\),这里是倒着进行字符串比较的意思

第\(k\)位的答案即为\(ans_1 \ xor \ ans_2\),这里讨论一下就可以得到了

考虑找到\(\le p\)位置的两个数组第一个不同的位置,然后比较大小

一个暴力的想法是,把每个不同的位置塞到set里面去,然后每次在set里面二分找一下位置,也是\(\log^2n\)的

考虑到每次修改的一个长为\(\log a\)连续的区间(这里实际上饶了一个大圈子)

所以把区间每\(\log a\)分一块,块也是有势能的,然后set里面一次赛一个块就可以了

复杂度就一个\(\log\)了


Code:

#include <cstdio>
#include <cctype>
#include <set>
#include <algorithm>
#define ll long long
using std::min;
const int SIZE=1<<21;
char ibuf[SIZE],*iS,*iT;
//#define gc() (iS==iT?(iT=(iS=ibuf)+fread(ibuf,1,SIZE,stdin),iS==iT?EOF:*iS++):*iS++)
#define gc() getchar()
template <class T>
void read(T &x)
{
int f=0;x=0;char c=gc();
while(!isdigit(c)) f|=c=='-',c=gc();
while(isdigit(c)) x=x*10+c-'0',c=gc();
if(f) x=-x;
}
const int N=1e6+10;
int n,t1,t2,t3;
int plu[N*30],dec[N*30],bel[N*30];
int edt[N],vis[N];
std::set <int> s;
std::set <int>::iterator it;
int qry(int p)
{
while(p%32!=0&&plu[p]==dec[p]) --p;
if(plu[p]!=dec[p])
{
if(plu[p]>dec[p]) return 1;
else return 0;
}
p=bel[p];
it=s.upper_bound(p);
if(s.empty()||it==s.begin()) return 1;
--it;
p=(*it)*32;
while(plu[p]==dec[p]) --p;
if(plu[p]>dec[p]) return 1;
else return 0;
}
int main()
{
read(n),read(t1),read(t2),read(t3);
int m=n*30+31,T=(m-1)/32+1;
for(int L,R=0,i=1;i<=T;i++)
{
L=R+1,R=i*32;
for(int j=L;j<=R;j++) bel[j]=i;
}
for(int op,a,b,k,i=1;i<=n;i++)
{
read(op);
if(op==1)
{
edt[0]=0;
read(a),read(b);
if(a>0)
{
for(int p,j=1;j<=30;j++)
if(a>>j-1&1)
{
p=j+b;
if(vis[bel[p]]!=i)
edt[++edt[0]]=bel[p],vis[bel[p]]=i;
while(plu[p])
{
plu[p++]=0;
if(vis[bel[p]]!=i)
edt[++edt[0]]=bel[p],vis[bel[p]]=i;
}
plu[p]=1;
}
}
else
{
a=-a;
for(int p,j=1;j<=30;j++)
if(a>>j-1&1)
{
p=j+b;
if(vis[bel[p]]!=i)
edt[++edt[0]]=bel[p],vis[bel[p]]=i;
while(dec[p])
{
dec[p++]=0;
if(vis[dec[p]]!=i)
edt[++edt[0]]=bel[p],vis[bel[p]]=i;
}
dec[p]=1;
}
}
for(int j=1;j<=edt[0];j++)
{
int L=(edt[j]-1)*32+1,R=edt[j]*32,flag=1;
for(int l=L;l<=R;l++)
if(plu[l]!=dec[l])
{
flag=0;
break;
}
if(flag)
{
if(s.find(edt[j])!=s.end()) s.erase(edt[j]);
}
else
s.insert(edt[j]);
}
}
else
{
read(k);
int p=k++;
int ans=0;
if(plu[k]==dec[k]) ans=1;
printf("%d\n",ans^qry(p));
}
}
return 0;
}

2019.5.31

最新文章

  1. 3. Longest Substring Without Repeating Characters(c++) 15ms
  2. 应用程序框架实战三十八:项目示例VS解决方案的创建(一)
  3. cookie session URL重写 与考试
  4. Android - ADB 的使用
  5. OpenCV 之 边缘检测
  6. [转]轻松解决oracle11g 空表不能exp导出的问题
  7. 穷举法破解 zebrone1.1
  8. SQLAlchemy 对象缓存和刷新
  9. 《Java程序设计》第六周学习总结
  10. 从AlphaGo谈通用型人工智能设计
  11. LA3353
  12. linux内核学习之四:进程切换简述
  13. 使用win8时删除自带的输入法的好方法
  14. 近期在调用 calendar.js 的时候出现中文乱码! 解决方式
  15. Enum变量值的Discretion
  16. 浩哥解析MyBatis源码(十)——Type类型模块之类型处理器
  17. Hadoop启动方式
  18. unity3D写一个hello world
  19. Python之re模块(结合具体业务)
  20. EF+Redis(StackExchange.Redis)实现分布式锁,自测可行

热门文章

  1. JS中算法之检索算法(查找算法)
  2. 二、制作BOM表格--物料表格--Bill of Materials
  3. 如何写一个bat文件,让他去执行某一个地方的bat文件
  4. delphi vlc 安装bug 处理编译错误"0" is an invalid value for the "DebugInformation" parameter of the "DCC"
  5. Django-自定义用户模型
  6. android ndk 编译 libevent
  7. python 装饰器 第八步:使用类来作为装饰器参数
  8. “希希敬敬对”队软件工程第九次作业-beta冲刺第六次随笔
  9. pandas认识
  10. Spring Boot 支持 HTTPS 如此简单,So easy!