【推导】【线段树】hdu5929 Basic Data Structure
题意:
维护一个栈,支持以下操作:
从当前栈顶加入一个0或者1;
从当前栈顶弹掉一个数;
将栈顶指针和栈底指针交换;
询问a[top] nand a[top-1] nand ... nand a[bottom]的值。
nand是这样定义的:
∙∙ 0 nand 0 = 1
∙∙ 0 nand 1 = 1
∙∙ 1 nand 0 = 1
∙∙ 1 nand 1 = 0
关键是要发现性质,任何数nand 0,都会变成1。反复nand上1的话,则值会交替变化。
所以假设当前栈顶在左侧,只需要找到最右侧的0的位置,然后按照其右侧1的数量的奇偶性输出零或者一即可(如果最右侧的0在最左端,则其右侧有奇数个1就输出1,否则输出零。如果最右侧的零不在最左端,则其右侧有奇数个1就输出零,否则输出1)。
栈顶在右侧的情况同理。
用线段树维护。
#include<cstdio>
#include<cstring>
using namespace std;
bool hav[400005<<2];
int a[400005];
int T,m,n,e[2];
void update(int p,int v,int rt,int l,int r){
if(l==r){
if(v==0){
hav[rt]=1;
}
else{
hav[rt]=0;
}
return;
}
int m=(l+r>>1);
if(p<=m){
update(p,v,rt<<1,l,m);
}
else{
update(p,v,rt<<1|1,m+1,r);
}
hav[rt]=(hav[rt<<1] || hav[rt<<1|1]);
}
int find1(int rt=1,int l=1,int r=n){
if(l==r){
return l;
}
int m=(l+r>>1);
if(hav[rt<<1]){
return find1(rt<<1,l,m);
}
else{
return find1(rt<<1|1,m+1,r);
}
}
int find2(int rt=1,int l=1,int r=n){
if(l==r){
return l;
}
int m=(l+r>>1);
if(hav[rt<<1|1]){
return find2(rt<<1|1,m+1,r);
}
else{
return find2(rt<<1,l,m);
}
}
int main(){
//freopen("h.in","r",stdin);
int x;
bool dir=0;
char op[10];
scanf("%d",&T);
memset(a,-1,sizeof(a));
for(int zu=1;zu<=T;++zu){
printf("Case #%d:\n",zu);
scanf("%d",&m);
e[0]=m;
e[1]=m+1;
n=2*m;
for(int i=1;i<=m;++i){
scanf("%s",op);
if(op[2]=='S'){
scanf("%d",&x);
if(!dir){
a[e[0]]=x;
update(e[0],x,1,1,n);
--e[0];
}
else{
a[e[1]]=x;
update(e[1],x,1,1,n);
++e[1];
}
}
else if(op[2]=='P'){
if(!dir){
++e[0];
a[e[0]]=-1;
update(e[0],-1,1,1,n);
}
else{
--e[1];
a[e[1]]=-1;
update(e[1],-1,1,1,n);
}
}
else if(op[2]=='V'){
dir^=1;
}
else{
if(e[0]==e[1]-1){
puts("Invalid.");
continue;
}
if(hav[1]){
if(!dir){
int p=find2();
if(p==e[0]+1){
puts((e[1]-p-1)%2==1 ? "1" : "0");
}
else{
puts((e[1]-p-1)%2==1 ? "0" : "1");
}
}
else{
int p=find1();
if(p==e[1]-1){
puts((p-e[0]-1)%2==1 ? "1" : "0");
}
else{
puts((p-e[0]-1)%2==1 ? "0" : "1");
}
}
}
else{
puts((e[1]-e[0]-1)%2==1 ? "1" : "0");
}
}
}
memset(a,-1,sizeof(int)*(n+1));
memset(hav,0,sizeof(bool)*(n*4+1));
}
return 0;
}
【推导】【线段树】hdu5929 Basic Data Structure的更多相关文章
- hdu-5929 Basic Data Structure(双端队列+模拟)
题目链接: Basic Data Structure Time Limit: 7000/3500 MS (Java/Others) Memory Limit: 65536/65536 K (Ja ...
- HDU 5929 Basic Data Structure 模拟
Basic Data Structure Time Limit: 7000/3500 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Oth ...
- HDU 5929 Basic Data Structure 【模拟】 (2016CCPC东北地区大学生程序设计竞赛)
Basic Data Structure Time Limit: 7000/3500 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Oth ...
- Basic Data Structure
Basic Data Structure Time Limit: 7000/3500 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Oth ...
- Basic Data Structure HDU - 5929 (这个模拟我要报警了)
Mr. Frog learned a basic data structure recently, which is called stack.There are some basic operati ...
- hdu 5929 Basic Data Structure
ゲート 分析: 这题看出来的地方就是这个是左结合的,不适用结合律,交换律. 所以想每次维护答案就不怎么可能了.比赛的时候一开始看成了异或,重读一遍题目了以后就一直去想了怎么维护答案...... 但是很 ...
- HDU 5929 Basic Data Structure(模拟 + 乱搞)题解
题意:给定一种二进制操作nand,为 0 nand 0 = 10 nand 1 = 1 1 nand 0 = 1 1 nand 1 = 0 现在要你模拟一个队列,实现PUSH x 往队头塞入x,POP ...
- E. Alice and the Unfair Game(推导线段树)
题:https://codeforces.com/contest/1236/problem/E 粗自:https://www.cnblogs.com/YSFAC/p/11715522.html #in ...
- OpenJudge cdqz/Data Structure Challenge 2 (Problem 5822) - 可持久化线段树
描述 给一个空数列,有M次操作,每次操作是以下三种之一: (1)在数列后加一个数 (2)求数列中某位置的值 (3)撤销掉最后进行的若干次操作(1和3) 输入 第一行一个正整数M. 接下来M行,每行开头 ...
随机推荐
- 面试整理(1):原生ajax
接到电话面试,有一些送分题答的不好,在这里整理一下 问题:原生ajax的工作流程是怎么样的? 老用封装好的工具,原生的ajax其实并不熟悉,今天复习一下.主要参考http://www.w3school ...
- NYOJ 2 括号配对问题 (模拟)
题目链接 描述 现在,有一行括号序列,请你检查这行括号是否配对. 输入 第一行输入一个数N(0<N<=100),表示有N组测试数据.后面的N行输入多组输入数据,每组输入数据都是一个字符串S ...
- LOW逼三人组(二)----选择排序算法
选择排序思路: 算法程序: def cal_time(func): # 装饰器 ,用来检测算法所执行的时间 def wrapper(*args,**kwargs): t1=time.time() re ...
- Go语言 8 反射
文章由作者马志国在博客园的原创,若转载请于明显处标记出处:http://www.cnblogs.com/mazg/ Go学习群:415660935 8.1概念和作用 Reflection(反射)在计算 ...
- 设计模式之Adapter
设计模式总共有23种模式这仅仅是为了一个目的:解耦+解耦+解耦...(高内聚低耦合满足开闭原则) 介绍: 将两个不兼容的类纠合在一起使用,属于结构型模式,也有人称它为wrapper(包装类). 包装类 ...
- windows安装linux虚拟机、修改apt源
记录一下windows安装虚拟机以及初始配置的一些坑. 安装VMware Workstation 直接百度搜索VMware,选择合适的版本下载: 按照一般软件的安装步骤安装VMware Worksta ...
- 如何在LINUX中开机、登陆、退出、定时、定期自动运行程序
1.开机启动时自动运行程序 Linux加载后, 它将初始化硬件和设备驱动, 然后运行第一个进程init.init根据配置文件继续引导过程,启动其它进程.通常情况下,修改放置在 /etc/rc或 /et ...
- 分布式队列Celery入门
Celery 是一个简单.灵活且可靠的,处理大量消息的分布式系统,并且提供维护这样一个系统的必需工具.它是一个专注于实时处理的任务队列,同时也支持任务调度.Celery 是语言无关的,虽然它是用 Py ...
- MySQL-索引工作原理及使用注意事项
1.为什么需要索引(Why is it needed)? 当数据保存在磁盘类存储介质上时,它是作为数据块存放.这些数据块是被当作一个整体来访问的,这样可以保证操作的原子性.硬盘数据块存储结构类似于链表 ...
- Python 类的名称空间和组合
一.Python类的名称空间 class Student(object): School = '北京大学' def __init__(self): pass stu1 = Student() stu1 ...