bzoj 1056 [HAOI2008]排名系统(1862 [Zjoi2006]GameZ游戏排名系统)
1056: [HAOI2008]排名系统
Time Limit: 10 Sec Memory Limit: 162 MB
Submit: 1854 Solved: 502
[Submit][Status][Discuss]
Description
排名系统通常要应付三种请求:上传一条新的得分记录、查询某个玩家的当前排名以及返回某个区段内的排名记录。当某个玩家上传自己最新的得分记录时,他原有的得分记录会被删除。为了减轻服务器负担,在返回某个区段内的排名记录时,最多返回10条记录。
Input
第
一行是一个整数n(n>=10)表示请求总数目。接下来n行,每行包含了一个请求。请求的具体格式如下: +Name Score
上传最新得分记录。Name表示玩家名字,由大写英文字母组成,不超过10个字符。Score为最多8位的正整数。 ?Name
查询玩家排名。该玩家的得分记录必定已经在前面上传。 ?Index
返回自第Index名开始的最多10名玩家名字。Index必定合法,即不小于1,也不大于当前有记录的玩家总数。
Output
对于?Name格式的请求,应输出一个整数表示该玩家当前的排名。对于?Index格式的请求,应在一行中依次输出从第Index名开始的最多10名玩家姓名,用一个空格分隔。
Sample Input
+ADAM 1000000 加入ADAM的得分记录
+BOB 1000000 加入BOB的得分记录
+TOM 2000000 加入TOM的得分记录
+CATHY 10000000 加入CATHY的得分记录
?TOM 输出TOM目前排名
?1 目前有记录的玩家总数为4,因此应输出第1名到第4名。
+DAM 100000 加入DAM的得分记录
+BOB 1200000 更新BOB的得分记录
+ADAM 900000 更新ADAM的得分记录(即使比原来的差)
+FRANK 12340000 加入FRANK的得分记录
+LEO 9000000 加入LEO的得分记录
+KAINE 9000000 加入KAINE的得分记录
+GRACE 8000000 加入GRACE的得分记录
+WALT 9000000 加入WALT的得分记录
+SANDY 8000000 加入SANDY的得分记录
+MICK 9000000 加入MICK的得分记录
+JACK 7320000 加入JACK的得分记录
?2 目前有记录的玩家总数为12,因此应输出第2名到第11名。
?5 输出第5名到第13名。
?KAINE 输出KAINE的排名
Sample Output
CATHY TOM ADAM BOB
CATHY LEO KAINE WALT MICK GRACE SANDY JACK TOM BOB
WALT MICK GRACE SANDY JACK TOM BOB ADAM DAM
4
HINT
100%数据满足N<=250000
【题解】
名次树+哈希表+处理相同结点。
总的思路就是利用名次树维护排名系统,利用插入时间clock标识每个节点,利用哈希表实现由name到插入时间t以及键值v的映射,这样就可以根据(v,t)在ranktree中实现查找。
其中?name操作可以如下完成:
int ans_cnt,ans[maxn];
void query(Node* o,int pos,int pre) {
if(o==NULL) return ;
int s=(o->ch[1]==NULL? 0:o->ch[1]->s);
int rank=pre+s+1;
if(rank>=pos && rank<=pos+9) {
query(o->ch[1],pos,pre);
ans[ans_cnt++]=o->c;
query(o->ch[0],pos,pre+s+1);
}
else {
if(rank<pos) query(o->ch[0],pos,pre+s+1);
else query(o->ch[1],pos,pre);
}
}
需要注意的有:
1)插入时将相同结点插入右子树,虽然可能会出现通过rotate旋转上来的情况,但其t的相对顺序不会发生改变,因此在遍历中如果遇到相等需要通过t判断所处子树的位置。
2) 行末无空格。
3) 不能以通过不旋转相同结点即破坏堆性质的方法处理相同,刚开始实(zuo)验(si)性试了一下,导致Treap直接退化。
4)bzoj上内存超了就是tle,另外如果要过1862我的代码还需要优化一下hash表以减少空间的浪费。
【代码】
#include<ctime>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<cstdlib>
using namespace std; const int maxn = +;
const int MOD = ; struct Node {
Node *ch[];
int v,r,s,c;
Node(int w,int t) :v(w),c(t) { ch[]=ch[]=NULL; s=; r=rand(); }
int cmp(int x) const {
if(v==x) return -;
return x<v? :;
}
int maintain() {
s=;
if(ch[]!=NULL) s+=ch[]->s;
if(ch[]!=NULL) s+=ch[]->s;
}
};
Node *root;
void rotate(Node* &o,int d) {
Node* k=o->ch[d^]; o->ch[d^]=k->ch[d],k->ch[d]=o;
o->maintain(),k->maintain(),o=k;
}
void insert(Node* &o,int x,int t) {
if(o==NULL) o=new Node(x,t);
else {
int d=x<=o->v? :;
insert(o->ch[d],x,t);
if(o->ch[d]->r > o->r) rotate(o,d^);
}
o->maintain();
}
void remove(Node* &o,int x,int t) {
int d=o->cmp(x);
if(d==- && o->c==t) {
Node *u=o;
if(o->ch[]!=NULL && o->ch[]!=NULL) {
int d2=o->ch[]->r > o->ch[]->r? :;
rotate(o,d2); remove(o->ch[d2],x,t);
}
else {
if(o->ch[]!=NULL) o=o->ch[];
else o=o->ch[]; //ch[0]==NULL && ch[1]==NULL
delete u;
}
}
else {
if(d==-) { if(t<o->c) d=; else d=; }
remove(o->ch[d],x,t);
}
if(o!=NULL) o->maintain();
}
int rank(Node* o,int x,int t) {
if(o==NULL) return -;
int s=o->ch[]==NULL? :o->ch[]->s;
int d=o->cmp(x);
if(d==- && t==o->c) return s+;
else {
if(d==-) { if(t<o->c) d=; else d=; }
if(d==) return s++rank(o->ch[],x,t);
else return rank(o->ch[],x,t);
}
} int n,score,iclock;
char tname[maxn][],s[];
int tt[MOD][],tv[MOD][],size[MOD]; char flag[MOD][][]; int ans_cnt,ans[maxn];
void query(Node* o,int pos,int pre) {
if(o==NULL) return ;
int s=(o->ch[]==NULL? :o->ch[]->s);
int rank=pre+s+;
if(rank>=pos && rank<=pos+) {
query(o->ch[],pos,pre);
ans[ans_cnt++]=o->c;
query(o->ch[],pos,pre+s+);
}
else {
if(rank<pos) query(o->ch[],pos,pre+s+);
else query(o->ch[],pos,pre);
}
}
int hash(char* s) {
int h=;
for(int i=;i<strlen(s);i++) h=h*+s[i]-'A'+,h%=MOD;
return h;
}
int find(char *s) {
int h=hash(s),i=;
while(i<size[h]) { if(strcmp(flag[h][i],s)==) return i; i++;
}
size[h]++; return i;
//cout<<"sz: "<<size[h]<<endl;
} int main() {
freopen("rank.in","r",stdin);
freopen("rank.out","w",stdout);
scanf("%d",&n);
while(n--) {
scanf("%s",s);
int h,r;
if(s[]=='+') {
scanf("%d",&score);
h=hash(s+),r=find(s+);
if(tv[h][r]) remove(root,tv[h][r],tt[h][r]);
insert(root,score,++iclock);
strcpy(tname[iclock],s);
strcpy(flag[h][r],s+);
tt[h][r]=iclock,tv[h][r]=score;
}
else {
if(isalpha(s[])) {
h=hash(s+),r=find(s+);
printf("%d\n",rank(root,tv[h][r],tt[h][r]));
}else {
int pos=;
for(int i=;i<strlen(s);i++) pos=pos*+s[i]-'';
ans_cnt=;
query(root,pos,);
for(int i=;i<ans_cnt;i++) {
printf("%s",tname[ans[i]]+);
if(i<ans_cnt-) putchar(' ');
}
putchar('\n');
}
}
}
return ;
}
bzoj 1056 [HAOI2008]排名系统(1862 [Zjoi2006]GameZ游戏排名系统)的更多相关文章
- bzoj1056: [HAOI2008]排名系统 && 1862: [Zjoi2006]GameZ游戏排名系统
hash 加上 平衡树(名次树). 这道题麻烦的地方就在于输入的是一个名字,所以需要hash. 这个hash用的是向后探查避免冲突,如果用类似前向星的方式避免冲突,比较难写,容易挂掉,但也速度快些. ...
- BZOJ 1862: [Zjoi2006]GameZ游戏排名系统 [treap hash]
1862: [Zjoi2006]GameZ游戏排名系统 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1318 Solved: 498[Submit][ ...
- 1056/1862. [ZJOI2006]GameZ游戏排名系统【平衡树-splay】
Description GameZ为他们最新推出的游戏开通了一个网站.世界各地的玩家都可以将自己的游戏得分上传到网站上.这样就可以看到自己在世界上的排名.得分越高,排名就越靠前.当两个玩家的名次相同时 ...
- 【BZOJ】1862: [Zjoi2006]GameZ游戏排名系统 & 1056: [HAOI2008]排名系统(treap+非常小心)
http://www.lydsy.com/JudgeOnline/problem.php?id=1862 http://www.lydsy.com/JudgeOnline/problem.php?id ...
- bzoj 1862: [Zjoi2006]GameZ游戏排名系统 & bzoj 1056: [HAOI2008]排名系统
傻叉了一晚上,把t打成x,然后这题神奇在于输出一段数,不足的不用输出,一开始我的是直接找没有后面就退,然后这样会格式错误囧……然后最后zj的还卡了下空间,于是不用string就过了……string毁一 ...
- 【pb_ds】bzoj1056 [HAOI2008]排名系统/bzoj1862 [Zjoi2006]GameZ游戏排名系统
STL裸题,线下AC,bzoj无限RE ing…… #include<cstdio> #include<cctype> #include<iostream> #in ...
- bzoj1056/1862 [Zjoi2006]GameZ游戏排名系统
题目链接:1,2 treap恶心题,不多说 #include<algorithm> #include<iostream> #include<cstdlib> #in ...
- BZOJ_1862_[Zjoi2006]GameZ游戏排名系统&&BZOJ_1056_[HAOI2008]排名系统_Splay
BZOJ_1862_[Zjoi2006]GameZ游戏排名系统&&BZOJ_1056_[HAOI2008]排名系统_Splay Description 排名系统通常要应付三种请求:上传 ...
- [HAOI2008]排名系统& [Zjoi2006]GameZ游戏排名系统
1056: [HAOI2008]排名系统 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2487 Solved: 711[Submit][Statu ...
随机推荐
- Swift: 基本操作符
这里只讲一下Swift中比较特殊的操作符,在其他语言中也存在操作符就不再讲了 Nil-Coalescing Operator: ?? The nil-coalescing operator (a ?? ...
- POJ 1985 Cow Marathon && POJ 1849 Two(树的直径)
树的直径:树上的最长简单路径. 求解的方法是bfs或者dfs.先找任意一点,bfs或者dfs找出离他最远的那个点,那么这个点一定是该树直径的一个端点,记录下该端点,继续bfs或者dfs出来离他最远的一 ...
- VB指针 与CopyMemory
体会ByVal和ByRef Dim k As Long CopyMemory ByVal VarPtr(k), 40000, 4 等同于k=40000:从保存常数40000(缺省ByRef)的临时变量 ...
- Visual Studio 2013 无法启动 IIS Express 的解决办法
关于 ASP.NET Web 开发服务器.本地 IIS和 IIS Express 的区别,请参见<VS2013无法启动IIS Express Web的解决办法>,此文章最后提到的部分,即是 ...
- $(document).ready(function(){}),$().ready(function(){})和$(function(){})三个有区别么
三者都是一样的,最完整的写法是:$(document).ready(function(){})ready() 函数仅能用于当前文档,因此无需选择器.所以document选择器可以不要,那么就可以写成: ...
- 【转】 iOS开发UI篇—控制器的View的创建
最近对view的周期等还不是非常清楚,就找到顶哥的文章,非常不错,就搬运过来了. 原文: http://www.cnblogs.com/wendingding/p/3770760.html 一.6种创 ...
- [转]Delphi中进行延时的4种方法
1.挂起,不占CPU sleep 2.不挂起,占cpu procedure Delay(msecs:integer); var FirstTickCount:longint; begin FirstT ...
- 实时错误 '91' :对象变量或with块变量未设置
大家这几天在做学生信息管理系统的时候,出现最多的应该就是这个问题了,“实时错误‘91’:对象变量或with块变量未设置”.如右图: 遇到这个问题,我们首先应该去参考MSDN,不过这时候MSDN似乎没有 ...
- css命名为何不推荐使用下划线_
一直习惯了在命名CSS样式名时使用下划线“_”做为单词的分隔符,这也是在写JS时惯用的写法. 用过CSS hack的朋友应该知道,用下划线命名也是一种hack,如使用“_style”这样的命名,可以让 ...
- 巧用C#做中间语言 实现Java调用.net DLL
本文将详细为大家介绍一个java调用.net DLL的方法,以实现特殊的客户的特殊要求:“在Java项目中必须使用其提供的用.net写的DLL加密机制!” 环境与工具: ◆.net framework ...