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 ...
随机推荐
- python05-09
一.lambda表达式 def f1(): return 123 f2 = lambda : 123 def f3 = (a1,a2): return a1+a2 f4 = lambda a1,a2 ...
- 数据库分表和分库的原理及基于thinkPHP的实现方法
为什么要分表,分库: 当我们的数据表数据量,訪问量非常大.或者是使用频繁的时候,一个数据表已经不能承受如此大的数据訪问和存储,所以,为了减轻数据库的负担,加快数据的存储,就须要将一张表分成多张,及将一 ...
- API Copy Big FIles
public class ApiCopyFile { private const int FO_COPY = 0x0002; private const int FOF_ALLOWUNDO = 0x0 ...
- IIS 配置 FTP 网站 H5 音频标签自定义样式修改以及添加播放控制事件
IIS 配置 FTP 网站 在 服务器管理器 的 Web服务器IIS 上安装 FTP 服务 在 IIS管理器 添加FTP网站 配置防火墙规则 说明:服务器环境是Windows Server 200 ...
- SDUT 3503 有两个正整数,求N!的K进制的位数
有两个正整数,求N!的K进制的位数 题目链接:action=showproblem&problemid=3503">http://sdutacm.org/sdutoj/prob ...
- mysql07---主从复制
mysql主从复制,replication,(可以一主多从,不可一从多主) 原理: 主从分离,最少2台服务器.主服务器里面的数据,要在从服务器里面都有一份. 把主服务器的所有insert,update ...
- JavaScript基础简要
JavaScript 引用外部js : <script src="2.js"type="text/javascript"></scri ...
- 【HDU 2196】 Computer
[题目链接] 点击打开链接 [算法] 我们知道,一棵树上离某个节点最远的节点,可能是经过它的祖先,再到那个祖先的某个孩子,或者,是它的那颗子树中,离它最远的一个节点,就不难想到以下算法 : 第一遍DF ...
- bzoj 3489 A simple rmq problem —— 主席树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3489 题解:http://www.itdaan.com/blog/2017/11/24/9b ...
- MultipartResolver实现文件上传功能
转自:https://www.jb51.net/article/142736.htm springMVC默认的解析器里面是没有加入对文件上传的解析的,,使用springmvc对文件上传的解析器来处理文 ...