hdu3436Queue-jumpers(线段树)
这题最不好求的一部分是rank部分 因为top每次都是把一个数放在队头 不会穿插在数组里 也就是说后面没有top过的一部分数 依旧保持有序
这样可以分为两部分来处理 比如 1 2 3 4 5 6 7 TOP 3 TOP 5 TOP 7后 会是这么一种状态 7 5 3 1 2 4 6 这样可以建立两颗线段树一个处理前面一个处理后面
对于N 值很大 肯定是需要离散化的 前面的那颗就可以用普通的求第k大 标记 求和来求出第几 以及x在哪个位置 后面的线段树的值 可以这么来表示 s[w] = num[i]-num[i-1]
这样对于询问rank x 就用线段树找第一个大于等于x的数。
虽然错了很多次,看见那个红红的AC还是很高兴的。。。
#include <iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<stdlib.h>
#include<vector>
#include<cmath>
#include<queue>
#include<set>
#include<map>
using namespace std;
#define N 100010
#define LL long long
#define INF 0xfffffff
const double eps = 1e-;
const double pi = acos(-1.0);
const double inf = ~0u>>;
struct node
{
char s[];
int d;
}p[N];
int s[N<<],o[N<<],ss[N<<],oo[N<<];
int po[N],fo[N],a[N];
bool vis[N];
map<int,int>f;
void up(int w)
{
s[w] = s[w<<]+s[w<<|];
}
void up1(int w)
{
ss[w] = ss[w<<]+ss[w<<|];
oo[w] = oo[w<<]+oo[w<<|];
}
void build(int l,int r,int w)
{
if(l==r)
{
s[w] = oo[w] = ;
if(fo[l]>fo[l-])
ss[w] = fo[l]-fo[l-];
return ;
}
int m= (l+r)>>;
build(l,m,w<<);
build(m+,r,w<<|);
up(w);
up1(w);
} void update(int pos,int k,int d,int l,int r,int w)
{
if(l==r)
{
s[w] = d;
if(d)
{
po[k] = l;
o[w] = fo[k];
// cout<<k<<" "<<fo[k]<<endl; }
return ;
}
int m = (l+r)>>;
if(pos>m) update(pos,k,d,m+,r,w<<|);
else update(pos,k,d,l,m,w<<);
up(w);
}
void update1(int k,int l,int r,int w)
{
if(l==r)
{
ss[w]--;
oo[w] = ;
return ;
}
int m = (l+r)>>;
if(k>m) update1(k,m+,r,w<<|);
else update1(k,l,m,w<<);
up1(w);
}
int query(int p,int l,int r,int w)
{
if(l==r)
{
return o[w];
}
int m = (l+r)>>;
if(p>s[w<<]) return query(p-s[w<<],m+,r,w<<|);
else return query(p,l,m,w<<);
}
int query1(int p,int l,int r,int w)
{//cout<<l<<" "<<r<<" "<<ss[w]<<" "<<p<<endl;
if(l==r)
{
//cout<<l<<endl;
if(vis[l])
return fo[l]--(ss[w]-p);
else
return fo[l]-(ss[w]-p);
}
int m = (l+r)>>;
if(p<=ss[w<<])
return query1(p,l,m,w<<);
else
return query1(p-ss[w<<],m+,r,w<<|);
}
int getsum(int a,int b,int l,int r,int w)
{
if(a<=l&&b>=r)
return s[w];
int m = (l+r)>>;
int re = ;
if(a<=m)
re+=getsum(a,b,l,m,w<<);
if(b>m)
re+=getsum(a,b,m+,r,w<<|);
return re;
}
int getsum1(int a,int b,int l,int r,int w)
{
if(a<=l&&b>=r)
return oo[w];
int m = (l+r)>>;
int re = ;
if(a<=m)
re+=getsum1(a,b,l,m,w<<);
if(b>m)
re+=getsum1(a,b,m+,r,w<<|);
return re;
}
int main()
{
int i,n,q,kk=,t;
cin>>t;
while(t--)
{
memset(vis,,sizeof(vis));
memset(fo,,sizeof(fo));
memset(o,,sizeof(o));
f.clear();
scanf("%d%d",&n,&q);
for(i = ; i < q; i++)
{
scanf("%s%d",p[i].s,&p[i].d);
a[i] = p[i].d;
}
sort(a,a+q);
int g = ;
for(i = ;i < q ;i++)
{
if(!f[a[i]])
{
g++;
f[a[i]] = g;
fo[g] = a[i];
//cout<<g<<" "<<a[i]<<endl;
}
}
g++;
f[n+] = g;
fo[g] = n+;
g = q+;
build(,g,);
int e = g+;
printf("Case %d:\n",++kk);
for(i = ; i < q ; i++)
{
int k = f[p[i].d];
if(strcmp(p[i].s,"Top")==)
{
e--;
if(vis[k])
update(po[k],k,,,g,);
else
update1(k,,g,);
update(e,k,,,g,);
vis[k]= ;
}
else if(strcmp(p[i].s,"Rank")==)
{
//int sum = getsum(1,g,1,g,1);
//cout<<s[1]<<endl;
//cout<<getsum1(1,2,1,g,1)<<endl;
if(p[i].d>s[])
{
int cnt = query1(p[i].d-s[],,g,);
printf("%d\n",cnt);
}
else
{
printf("%d\n",query(p[i].d,,g,));
}
}
else
{
if(vis[k])
{
int cnt = getsum(,po[k],,g,);
//cout<<po[k]<<endl;
printf("%d\n",cnt);
}
else
{
if(k==g)
cout<<p[i].d<<endl;
else
{
int cnt = getsum1(k+,g,,g,);
printf("%d\n",cnt+p[i].d);
}
}
}
}
}
return ;
}
hdu3436Queue-jumpers(线段树)的更多相关文章
- bzoj3932--可持久化线段树
		
题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...
 - codevs 1082 线段树练习 3(区间维护)
		
codevs 1082 线段树练习 3 时间限制: 3 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...
 - codevs 1576 最长上升子序列的线段树优化
		
题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...
 - codevs 1080 线段树点修改
		
先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...
 - codevs 1082 线段树区间求和
		
codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...
 - PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树
		
#44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...
 - CF719E(线段树+矩阵快速幂)
		
题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ...
 - 【BZOJ-3779】重组病毒      LinkCutTree + 线段树 + DFS序
		
3779: 重组病毒 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 224 Solved: 95[Submit][Status][Discuss] ...
 - 【BZOJ-3673&3674】可持久化并查集       可持久化线段树 + 并查集
		
3673: 可持久化并查集 by zky Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 1878 Solved: 846[Submit][Status ...
 - 【BZOJ-2653】middle      可持久化线段树 + 二分
		
2653: middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1298 Solved: 734[Submit][Status][Discu ...
 
随机推荐
- MySQL多实例配置(一)
			
MySQL数据库的集中化运维,能够通过在一台MySQL数据库服务器上,部署多个MySQL实例.该功能是通过mysqld_multi来实现.mysqld_multi用于管理多个mysqld的服务进程,这 ...
 - 【iOS开发】-NSString的扩展使用
			
第一:基本数据类型与字符串转换 //基本数据类型(int float,double char) 1)基本数据类型->NSString //1.int类型换换成字符串 int a = 88; NS ...
 - Android Studio keymap到Eclipse后,查找下一个同样变量快捷键Ctrl+K失效
			
注:升级到0.8的版本号以后.这个快捷键能够使了,只是另一个bug,假设你用了Ctrl+F先去查找了其它的东东,再使这个快捷键去定位另外一个变量可能偶尔会不灵,不灵的话还是能够用我以下的方式来让Ctr ...
 - [IT学习]Python如何处理异常特殊字符
			
欢迎访问www.cnblogs.com/viphhs A byte of Python<输入与输出>一节中有一个处理回文的小例子(io_input.py).作者留了个思考题. 如何将标点去 ...
 - oracle连接串的一种写法
			
我在.NET项目里访问oracle,向来是规规矩矩地这样写: DATA SOURCE=PDBGZFBC;PASSWORD=test;PERSIST SECURITY INFO=True;USER ID ...
 - C#中,变量前的@符号
			
看别人写的C#代码,发现有变量前带@,啥意思? string @namespace = "EnterpriseServerBase.WebService.DynamicWebCalling& ...
 - cygwin安装sshd服务(win7)Error installing a service: OpenSCManager: Win32 error 5:
			
Error installing a service: OpenSCManager: Win32 error 5: 出现这个问题的解决办法:win7系统管理员运行Cygwin软件 ...
 - session与cookie的理解
			
噶,nation同志又给我布置任务了~~是隔壁家老王提到多次的,session和cookie的运行机制问题~~于是我真的意识到一个问题,程序猿都是一个德行! 1. cookie概述 cookie 常用 ...
 - action 与 action 之间的跳转
			
这个要看你的两个action是不是在通一个package下了 <!-- 相同package下调用其它action --><action name="loginUser&qu ...
 - linux内存管理之uboot第一步
			
在进入讲解linux内存管理的kernel阶段以前,了解一下uboot阶段是如何准备好内存物理设备的,这是非常有意义的.通常进入到linux内核阶段之后,对内存芯片的物理特性寄存器访问是比较少的,强调 ...