bzoj 3685
线段树
方法一:
值域线段树,递归去写的,每次节点存出现次数.
对于几个操作, 1,2 直接加减就好 ; 3,4 操作贪心往某一个方向找 .7也很简单,主要说前驱后继怎么找。我是先找这个数第几小,根据相应关系得出我们需要输出第几小。然后就写了一下 TLE。详见代码一;
解决办法
加入输入外挂 .... 过了.详见代码二。
方法二:
参考了zkw,原来就是递归改为循环,再用到满二叉树性质.有些能够O(1) 求出.
具体做法:先确定范围,让二叉树最后一层节点数>=n.并找到最后一层节点编号 L-R .和前面有sz个节点. 增删我们从 sz+x这个节点开始加上一个数,之后除以2.
再写一个查询最大最小.那个函数传入的是节点编号.所以他可以查询某个子树的max,min.结合到前驱后继。比如查询x前驱,我就从x那个最底层点开始,看我是右儿子?如果是,看左儿子节点有数?有查询那颗子树的max,即为答案,反之也是一样的。最后一个操作直接来。sz+x。对于下标处理原题 [0,n-1] . 线段树 是1开始所以减一莫要忘记.
具体写法,参考代码三。
//代码一
#include <stdio.h>
const int maxn=1e6+10;
int T[maxn*4];
void ins(int i,int l,int r,int x,int val)// 1 2
{
if(l==r)
{
T[i]=val;
return ;
}
int mid=(l+r)>>1;
if(x<=mid) ins(i<<1,l,mid,x,val);
else ins(i<<1|1,mid+1,r,x,val);
T[i]=T[i<<1]+T[i<<1|1];
}
int fmax(int i,int l,int r)
{
if(T[i]==0) return -1;
if(l==r) return l;
int mid=(l+r)>>1;
if(T[i<<1|1]) return fmax(i<<1|1,mid+1,r);
else return fmax(i<<1,l,mid);
}
int fmin(int i,int l,int r)
{
if(T[i]==0) return -1;
if(l==r) return l;
int mid=(l+r)>>1;
if(T[i<<1]) return fmin(i<<1,l,mid);
else return fmin(i<<1|1,mid+1,r);
}
int fx(int i,int l,int r,int x)
{
if(l==r) return T[i]?1:-1;
int mid=(l+r)>>1;
if(x<=mid) return fx(i<<1,l,mid,x);
else return fx(i<<1|1,mid+1,r,x);
}
bool flag;
int kth(int i,int l,int r,int x)
{
if(l==r)
{
flag=T[i];
return T[i];
}
int mid=(l+r)>>1;
if(x<=mid) return kth(i<<1,l,mid,x);
else return T[i<<1]+kth(i<<1|1,mid+1,r,x);
}
int fkth(int i,int l,int r,int k)
{
if(l==r)
{
if(T[i]) return l;
else return -1;
}
int mid=(l+r)>>1;
int sum=T[i<<1];
if(sum>=k) return fkth(i<<1,l,mid,k);
else return fkth(i<<1|1,mid+1,r,k-sum);
}
int main()
{
int n,m;
scanf("%d%d",&n,&m);
while(m--)
{
int op,x;
scanf("%d",&op);
if(op==3||op==4) ;
else scanf("%d",&x);
if(op==1) ins(1,-1,n+1,x,1);
else if(op==2) ins(1,-1,n+1,x,0);
else if(op==3) printf("%d\n",fmin(1,-1,n+1));
else if(op==4) printf("%d\n",fmax(1,-1,n+1));
else if(op==5)
{
int k=kth(1,-1,n+1,x);
//printf("k=%d flag=%d\n",k,flag?1:0);
if(flag) k--;
printf("%d\n",fkth(1,-1,n+1,k));
}
else if(op==6)
{
int k=kth(1,-1,n+1,x);
k++;
printf("%d\n",fkth(1,-1,n+1,k));
}
else printf("%d\n",fx(1,-1,n+1,x));
}
return 0;
}
/*
10 100
1 2
1 4
1 6
1 8
1 10
*/
//代码二
#include <stdio.h>
const int maxn=1e6+10;
inline char nc()
{
static char buf[100000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;
}
inline int sc()
{
char ch=nc();
int sum=0;
while(!(ch>='0'&&ch<='9'))ch=nc();
while(ch>='0'&&ch<='9')sum=sum*10+ch-48,ch=nc();
return sum;
}
int T[maxn*4];
void ins(int i,int l,int r,int x,int val)// 1 2
{
if(l==r)
{
T[i]=val;
return ;
}
int mid=(l+r)>>1;
if(x<=mid) ins(i<<1,l,mid,x,val);
else ins(i<<1|1,mid+1,r,x,val);
T[i]=T[i<<1]+T[i<<1|1];
}
int fmax(int i,int l,int r)
{
if(T[i]==0) return -1;
if(l==r) return l;
int mid=(l+r)>>1;
if(T[i<<1|1]) return fmax(i<<1|1,mid+1,r);
else return fmax(i<<1,l,mid);
}
int fmin(int i,int l,int r)
{
if(T[i]==0) return -1;
if(l==r) return l;
int mid=(l+r)>>1;
if(T[i<<1]) return fmin(i<<1,l,mid);
else return fmin(i<<1|1,mid+1,r);
}
int fx(int i,int l,int r,int x)
{
if(l==r) return T[i]?1:-1;
int mid=(l+r)>>1;
if(x<=mid) return fx(i<<1,l,mid,x);
else return fx(i<<1|1,mid+1,r,x);
}
bool flag;
int kth(int i,int l,int r,int x)
{
if(l==r)
{
flag=T[i];
return T[i];
}
int mid=(l+r)>>1;
if(x<=mid) return kth(i<<1,l,mid,x);
else return T[i<<1]+kth(i<<1|1,mid+1,r,x);
}
int fkth(int i,int l,int r,int k)
{
if(l==r)
{
if(T[i]) return l;
else return -1;
}
int mid=(l+r)>>1;
int sum=T[i<<1];
if(sum>=k) return fkth(i<<1,l,mid,k);
else return fkth(i<<1|1,mid+1,r,k-sum);
}
int main()
{
int n,m;
scanf("%d%d",&n,&m);
while(m--)
{
int op,x;
op=sc();
if(op==3||op==4) ;
else x=sc();
if(op==1) ins(1,-1,n+1,x,1);
else if(op==2) ins(1,-1,n+1,x,0);
else if(op==3) printf("%d\n",fmin(1,-1,n+1));
else if(op==4) printf("%d\n",fmax(1,-1,n+1));
else if(op==5)
{
int k=kth(1,-1,n+1,x);
//printf("k=%d flag=%d\n",k,flag?1:0);
if(flag) k--;
printf("%d\n",fkth(1,-1,n+1,k));
}
else if(op==6)
{
int k=kth(1,-1,n+1,x);
k++;
printf("%d\n",fkth(1,-1,n+1,k));
}
else printf("%d\n",fx(1,-1,n+1,x));
}
return 0;
}
/*
10 100
1 2
1 4
1 6
1 8
1 10
*/
//代码三
#include <stdio.h>
#include <stdlib.h>
#include <algorithm>
const int maxn=1e6+10;
int T[maxn*4],sz,L,R;
void ins(int i,int val)
{
i=i+sz;
if(T[i]==1&&val==1) return ;
if(T[i]==0&&val==-1) return ;
for(;i>0;i>>=1) T[i]+=val;
}
int fmax(int i)
{
if(T[i]==0) return 0;
while(i<=sz)
{
if(T[i<<1|1]) i=i<<1|1;
else i=i<<1;
}
return i-sz;
}
int fmin(int i)
{
if(T[i]==0) return 0;
while(i<=sz)
{
if(T[i<<1]) i=i<<1;
else i=i<<1|1;
}
return i-sz;
}
int fpre(int i)
{
i=i+sz;
for(;i!=1;i>>=1)
if((i&1)&&T[i^1]) return fmax(i^1);
return 0;
}
int fsuc(int i)
{
i=i+sz;
for(;i!=1;i>>=1)
if(!(i&1)&&T[i^1]) return fmin(i^1);
return 0;
}
int main()
{
int n,m;
scanf("%d%d",&n,&m);
sz=0;
for(int i=1;i<n;i=(i<<1))
{
sz+=i;
L=sz+1;
R=sz+(i<<1);
}
while(m--)
{
int op,x;
scanf("%d",&op);
if(op==3||op==4);
else scanf("%d",&x);
x++;
if(op==1) ins(x,1);
else if(op==2) ins(x,-1);
else if(op==3) printf("%d\n",fmin(1)-1);
else if(op==4) printf("%d\n",fmax(1)-1);
else if(op==5) printf("%d\n",fpre(x)-1);
else if(op==6) printf("%d\n",fsuc(x)-1);
else printf("%d\n",T[sz+x]?1:-1);
}
return 0;
}
bzoj 3685的更多相关文章
- BZOJ 3685: 普通van Emde Boas树( 线段树 )
建颗权值线段树就行了...连离散化都不用... 没加读入优化就TLE, 加了就A掉了...而且还快了接近1/4.... ---------------------------------------- ...
- bzoj 3685: 普通van Emde Boas树
3685: 普通van Emde Boas树 Description 设计数据结构支持:1 x 若x不存在,插入x2 x 若x存在,删除x3 输出当前最小值,若不存在输出-14 输出当 ...
- 【模板】BZOJ 3685: 普通van Emde Boas树——Treap
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3685 据说神犇都是用zkw线段树水过的啊... 我蒟蒻只会写treap,加了fread之后8 ...
- BZOJ 2127: happiness [最小割]
2127: happiness Time Limit: 51 Sec Memory Limit: 259 MBSubmit: 1815 Solved: 878[Submit][Status][Di ...
- BZOJ 3275: Number
3275: Number Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 874 Solved: 371[Submit][Status][Discus ...
- BZOJ 2879: [Noi2012]美食节
2879: [Noi2012]美食节 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1834 Solved: 969[Submit][Status] ...
- bzoj 4610 Ceiling Functi
bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...
- BZOJ 题目整理
bzoj 500题纪念 总结一发题目吧,挑几道题整理一下,(方便拖板子) 1039:每条线段与前一条线段之间的长度的比例和夹角不会因平移.旋转.放缩而改变,所以将每条轨迹改为比例和夹角的序列,复制一份 ...
- 【sdoi2013】森林 BZOJ 3123
Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负整数 ...
随机推荐
- 重读金典------高质量C编程指南(林锐)-------第三章 命名规则
3.1 共性规则 规则:标识符应该直观且可以拼读,可进行英语翻译. 规则:标识符的长度需要控制好,不应该太长. 规则:命名规则应该同操作系统或者开发工具等保持一致,比如大小写混用.AddChar ...
- redis实现訪问频次限制的几种方式
结合上一篇文章<redis在学生抢房应用中的实践小结>中提及的用redis实现DDOS设计时遇到的expire的坑.事实上,redis官网中对incr命令的介绍中已经有关于怎样用redis ...
- int a; int* a; int** a; int (*a)[]; int (*a)(int)
a) int a;表示一个内存空间,这个空间用来存放一个整数(int):b) int* a;表示一个内存空间,这个空间用来存放一个指针,这个指针指向一个存放整数的空间,即a)中提到的空间:c) int ...
- Java 嵌套类和内部类演示样例<二>
嵌套类(nested class)是一个在还有一个类或接口内部声明的类. 嵌套类分为两种:静态内部类(static inner class)和非静态嵌套类(non-static nested clas ...
- oracle函数中lead,lag,over,partition by 的使用
lead,lag函数的分析 http://blog.csdn.net/mazongqiang/article/details/7621328 举例如下: SQL> select * from ...
- Codeforces 467D Fedor and Essay bfs
题目链接: 题意: 给定n个单词. 以下有m个替换方式.左边的单词能变成右边的单词. 替换随意次后使得最后字母r个数最少,在r最少的情况下单词总长度最短 输出字母r的个数和单词长度. 思路: 我们觉得 ...
- Java学习之路 第四篇 oop和class (面向对象和类)
本人水平有限,创作本文是为了记录学习和帮助初学者学习,欢迎指正和补充 一.面向对象编程的设计概述 很多同学都在学校学了电脑的编程,现在的书籍大部分都是oop面向对象编程,一个很抽象的的名字,比较难以理 ...
- alibaba canal安装笔记
canal是alibaba开源的基于mysql binlog解析工具,可利用它实现mysql增量订阅/消费,典型的应用场景如下图: 利用canal,可以将mysql的数据变化,通过解析binlog,投 ...
- Can a GridView have a footer and header just like ListView?
Aquick question: In ListView I use this code: list.addHeaderView(headerView); How to deal with it wh ...
- python 基础 7.5 commands 模块
一. commands 模块 1.commands 模块只使用与linxu 的shell 模式下 在我们平时码字时,经常需要调用系统脚本或者系统命令来解决很多问题,接下来,我们就介绍给大家一个很好 ...