BZOJ1861:[ZJOI2006]Book书架

Description

小T有一个很大的书柜。这个书柜的构造有些独特,即书柜里的书是从上至下堆放成一列。她用1到n的正整数给每本书都编了号。 小T在看书的时候,每次取出一本书,看完后放回书柜然后再拿下一本。由于这些书太有吸引力了,所以她看完后常常会忘记原来是放在书柜的什么位置。不过小T的记忆力是非常好的,所以每次放书的时候至少能够将那本书放在拿出来时的位置附近,比如说她拿的时候这本书上面有X本书,那么放回去时这本书上面就只可能有X-1、X或X+1本书。 当然也有特殊情况,比如在看书的时候突然电话响了或者有朋友来访。这时候粗心的小T会随手把书放在书柜里所有书的最上面或者最下面,然后转身离开。 久而久之,小T的书柜里的书的顺序就会越来越乱,找到特定的编号的书就变得越来越困难。于是她想请你帮她编写一个图书管理程序,处理她看书时的一些操作,以及回答她的两个提问:(1)编号为X的书在书柜的什么位置;(2)从上到下第i本书的编号是多少。

Input

第一行有两个数n,m,分别表示书的个数以及命令的条数;第二行为n个正整数:第i个数表示初始时从上至下第i个位置放置的书的编号;第三行到m+2行,每行一条命令。命令有5种形式: 1. Top S——表示把编号为S的书房在最上面。 2. Bottom S——表示把编号为S的书房在最下面。 3. Insert S T——T∈{-1,0,1},若编号为S的书上面有X本书,则这条命令表示把这本书放回去后它的上面有X+T本书; 4. Ask S——询问编号为S的书的上面目前有多少本书。 5. Query S——询问从上面数起的第S本书的编号。

Output

对于每一条Ask或Query语句你应该输出一行,一个数,代表询问的答案。

Sample Input

10 10
1 3 2 7 5 8 10 4 9 6
Query 3
Top 5
Ask 6
Bottom 3
Ask 3
Top 6
Insert 4 -1
Query 5
Query 2
Ask 2

Sample Output

2
9
9
7
5
3

HINT

100%的数据,n,m < = 80000

 

题解:

题目要求支持在数列中移动、询问元素,平衡树毫无疑问可以解决这种问题。但是,区间线段树也可以解决这个问题。

每个线段树节点记录对应的区间里有多少本书,则询问操作变成了单点访问、区间求和。

因为该题的操作只有交换相邻、把某个元素放到最前、把某个元素放到最后三种,所有书所占用的区间左右段点最多向外移动m,可以开n+2*m的区间来解决。

注意记录每本书在哪个位置、某个位置有什么书,移动时记得更新。

 

代码:

var
  i,j,k,l,y,n,m,ls,rs,cnt:longint;
  t:array[0..500000,-2..2]of longint;
  wz,bh:array[0..250000]of longint;
  ch,ch2:char;
procedure build(l,r,fa:longint);
var x:longint;
begin
  inc(cnt); x:=cnt; t[x,1]:=l; t[x,2]:=r;
  if t[x,1]=t[fa,1] then t[fa,-1]:=x else t[fa,-2]:=x;
  if l=r then
  begin
    if bh[l]>0 then t[x,0]:=1;
    exit;
  end;
  build(l,(l+r)div 2,x); build((l+r)div 2+1,r,x);
  t[x,0]:=t[t[x,-1],0]+t[t[x,-2],0];
end;
procedure work(x,y,z:longint);
begin
  t[x,0]:=t[x,0]+z;
  if t[x,1]=t[x,2] then exit;
  if y<=(t[x,1]+t[x,2])div 2 then work(t[x,-1],y,z)
  else work(t[x,-2],y,z);
end;
function qq(x,l,r:longint):longint;
var ll,rr:longint;
begin
  if(t[x,1]=l)and(t[x,2]=r)then exit(t[x,0]);
  ll:=t[x,1]; rr:=t[x,2];
  if r<=(ll+rr)div 2 then exit(qq(t[x,-1],l,r))
  else if l>(ll+rr)div 2 then exit(qq(t[x,-2],l,r))else
  exit(qq(t[x,-1],l,(ll+rr)div 2)+qq(t[x,-2],(ll+rr)div 2+1,r));
end;
function qq2(y:longint):longint;
var k,l:longint;
begin
  k:=1;
  while t[k,1]<>t[k,2] do
  begin
    l:=t[k,-1];
    if t[l,0]>=y then k:=l
    else begin y:=y-t[l,0]; k:=t[k,-2]; end;
  end;
  exit(bh[t[k,1]]);
end;
begin
  readln(n,m);
  for i:=1 to n do
  begin
    read(j);
    bh[i+80000]:=j;
    wz[j]:=i+80000;
  end;
  readln;
  ls:=80000; rs:=80000+n+1;
  build(0,80000+n+80000,0);
  for i:=1 to m do
  begin
    read(ch);
    read(ch2);
    while ch2<>' ' do read(ch2);
    if ch='Q' then
    begin
      readln(j); writeln(qq2(j));
    end else
    if ch='T' then
    begin
      readln(j);
      work(1,wz[j],-1);
      work(1,ls,1); bh[wz[j]]:=0; wz[j]:=ls; bh[ls]:=j; dec(ls);
    end else
    if ch='B' then
    begin
      readln(j);
      work(1,wz[j],-1);
      work(1,rs,1); bh[wz[j]]:=0; wz[j]:=rs; bh[rs]:=j; inc(rs);
    end else
    if ch='A' then
    begin
      readln(j);
      writeln(qq(1,0,wz[j]-1));
    end else
    begin
      readln(j,k);
      if k=0 then continue;
      l:=qq(1,0,wz[j]);
      l:=qq2(l+k);
      y:=wz[j]; wz[j]:=wz[l]; wz[l]:=y;
      y:=bh[wz[j]]; bh[wz[j]]:=bh[wz[l]]; bh[wz[l]]:=y;
    end;
  end;
end.
View Code
posted @ 2017-01-08 18:53  GhoStreach  阅读(317)  评论(0编辑  收藏  举报