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 ...
随机推荐
- 使用正則表達式的格式化与高亮显示json字符串
使用正則表達式的格式化与高亮显示json字符串 json字符串非常实用,有时候一些后台接口返回的信息是字符串格式的,可读性非常差,这个时候要是有个能够格式化并高亮显示json串的方法那就好多了,以下看 ...
- 大数据处理之道 (htmlparser获取数据<一>)
一:简单介绍 (1)HTML Parser是一个用于解析Html的Java的库.可採用线性或嵌套两种方式.主要用于网页的转换或提取,他有一些特性:过滤器filter,遍历器visitors,通常的标签 ...
- 适合初学C语言是练习的代码
作为一个小白,自己学C的时候就想找些代码练练手,就整理了一些. 1.最大公约数和最小公倍数 # include <stdio.h> int main(void) { int i, ...
- myeclipse配置hadoop开发环境
1.安装Hadoop开发插件 hadoop安装包contrib/目录下有个插件hadoop-0.20.2-eclipse-plugin.jar,拷贝到myeclipse根目录下/dropins目录下. ...
- HttpsURLConnection 安全传输(HTTPS--Secure Hypertext Transfer Protocol-安全超文本传输协议)
HttpsURLConnection 扩展 HttpURLConnection,支持各种特定于 https 功能.此类使用 HostnameVerifier 和 SSLSocketFactory.为这 ...
- linux怎么区别文本文件和二进制文件
linux的文本文件与二进制文件的区分与windows的区分是相同的!说到底计算机存储的文件都是以二进制形式存储的,但是区别是,习惯上认为: (1).文本文件 文本文件是包含用户可读信息的文件.这些文 ...
- C#温度报警
using System; using System.Collections.Generic; using System.Text; using System.Collections; using S ...
- bzoj2038 小Z的袜子(hose)——莫队算法
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2038 就是莫队算法: 先写了个分块,惨WA: #include<iostream> ...
- Java-Runoob-高级教程-实例-字符串:14. Java 实例 - 连接字符串
ylbtech-Java-Runoob-高级教程-实例-字符串:14. Java 实例 - 连接字符串 1.返回顶部 1. Java 实例 - 连接字符串 Java 实例 以下实例演示了通过 &qu ...
- OC:基础总结
OC面向对象的编程语言思想 类与对象.继承与实例化.属性点语法.内存管理.字符串.可见度. 类是一组具有相同特征和行为的事物的抽象 OC的与C相比所具有的新的特点: 定义新的类.类的实例和方法.方法的 ...