线段树

方法一:

值域线段树,递归去写的,每次节点存出现次数.

对于几个操作, 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的更多相关文章

  1. BZOJ 3685: 普通van Emde Boas树( 线段树 )

    建颗权值线段树就行了...连离散化都不用... 没加读入优化就TLE, 加了就A掉了...而且还快了接近1/4.... ---------------------------------------- ...

  2. bzoj 3685: 普通van Emde Boas树

    3685: 普通van Emde Boas树 Description 设计数据结构支持:1 x  若x不存在,插入x2 x  若x存在,删除x3    输出当前最小值,若不存在输出-14    输出当 ...

  3. 【模板】BZOJ 3685: 普通van Emde Boas树——Treap

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3685 据说神犇都是用zkw线段树水过的啊... 我蒟蒻只会写treap,加了fread之后8 ...

  4. BZOJ 2127: happiness [最小割]

    2127: happiness Time Limit: 51 Sec  Memory Limit: 259 MBSubmit: 1815  Solved: 878[Submit][Status][Di ...

  5. BZOJ 3275: Number

    3275: Number Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 874  Solved: 371[Submit][Status][Discus ...

  6. BZOJ 2879: [Noi2012]美食节

    2879: [Noi2012]美食节 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1834  Solved: 969[Submit][Status] ...

  7. bzoj 4610 Ceiling Functi

    bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...

  8. BZOJ 题目整理

    bzoj 500题纪念 总结一发题目吧,挑几道题整理一下,(方便拖板子) 1039:每条线段与前一条线段之间的长度的比例和夹角不会因平移.旋转.放缩而改变,所以将每条轨迹改为比例和夹角的序列,复制一份 ...

  9. 【sdoi2013】森林 BZOJ 3123

    Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负整数 ...

随机推荐

  1. PL/SQL Developer 和 instantclient客户端安装配置

    PL/SQL Developer 和 instantclient客户端安装配置 oracle的安装我就不写了,不会安装的网上随便找一个教程就能装上,安装起来比較简单.可是,PL/SQL Develop ...

  2. 在 Selenium 中让 PhantomJS 执行它的 API

    from selenium import webdriver driver = webdriver.PhantomJS() script = "var page = this; page.o ...

  3. Windows下ADB使用相关问题

    Windows下ADB使用相关问题 适用环境: 在Windows XP.WIN7下均可按本文操作进行.WIN8下没有进行实验.但操作设置大致同样.除了第4步.adb_usb.ini的位置可能有所不同以 ...

  4. netty 对 http 的实现

    netty的http协议栈无论是性能还是可靠性,都表现优异,非常适合在非web容器场景 下应用,相比于tomcat.jetty等web容器,它更轻量.小巧.灵活性和定制性也好: 总结:只要实现了htt ...

  5. 淘宝数据库OceanBase SQL编译器部分 源代码阅读--解析SQL语法树

    OceanBase是阿里巴巴集团自主研发的可扩展的关系型数据库,实现了跨行跨表的事务,支持数千亿条记录.数百TB数据上的SQL操作. 在阿里巴巴集团下,OceanBase数据库支持了多个重要业务的数据 ...

  6. jdbc 模板 连接

    package itcast; import java.sql.Connection;import java.sql.DriverManager;import java.sql.ResultSet;i ...

  7. Unity3D 动态地创建识别图

    前面介绍了EasyAR的单图识别,它是提前在Unity设置好图片路径的,那么如果我们的图片是存储在服务器上的,那么我们肯定不能直接把服务的图片地址填上去了.这个时候我们可以动态地创建识别图.步骤如下: ...

  8. 【BZOJ1124】[POI2008]枪战Maf 贪心+思路题

    [BZOJ1124][POI2008]枪战Maf Description 有n个人,每个人手里有一把手枪.一开始所有人都选定一个人瞄准(有可能瞄准自己).然后他们按某个顺序开枪,且任意时刻只有一个人开 ...

  9. EasyPlayerPro Windows播放器本地快照抓拍截图功能实现方法

    背景描述 作为一个播放器,截图功能必不可少; 下面主要记录一下截图功能的实现: 实现流程 将解码后的帧进行格式转换(目标格式为RGB24); 采用独立的线程进行截图处理; 截图可保存为BMP或JPG两 ...

  10. python文件读写方式

    window下换行\r\n linux.unix.mac下都是\n - 以二进制的形式wb写入,同样以二进制的方式读取rb ``` f = open('file name','wb') f.write ...