1507: [NOI2003]Editor
Time Limit: 5 Sec Memory Limit: 162 MBSubmit: 2685 Solved: 1066
[ Submit][ Status][ Discuss]
Description
Input
输入文件editor.in的第一行是指令条数t,以下是需要执行的t个操作。其中: 为了使输入文件便于阅读,Insert操作的字符串中可能会插入一些回车符,请忽略掉它们(如果难以理解这句话,可以参考样例)。 除了回车符之外,输入文件的所有字符的ASCII码都在闭区间[32, 126]内。且行尾没有空格。 这里我们有如下假定: MOVE操作不超过50000个,INSERT和DELETE操作的总个数不超过4000,PREV和NEXT操作的总个数不超过200000。 所有INSERT插入的字符数之和不超过2M(1M=1024*1024),正确的输出文件长度不超过3M字节。 DELETE操作和GET操作执行时光标后必然有足够的字符。MOVE、PREV、NEXT操作必然不会试图把光标移动到非法位置。 输入文件没有错误。 对C++选手的提示:经测试,最大的测试数据使用fstream进行输入有可能会比使用stdio慢约1秒。
Output
输出文件editor.out的每行依次对应输入文件中每条GET指令的输出。
Sample Input
15
Insert 26
abcdefghijklmnop
qrstuv wxy
Move 15
Delete 11
Move 5
Insert 1
^
Next
Insert 1
_
Next
Next
Insert 4
.\/.
Get 4
Prev
Insert 1
^
Move 0
Get 22
Insert 26
abcdefghijklmnop
qrstuv wxy
Move 15
Delete 11
Move 5
Insert 1
^
Next
Insert 1
_
Next
Next
Insert 4
.\/.
Get 4
Prev
Insert 1
^
Move 0
Get 22
Sample Output
.\/.
abcde^_^f.\/.ghijklmno
abcde^_^f.\/.ghijklmno
HINT
Source
这道题和bzoj1269文本编辑器很像,方法都是Splay维护区间。
只需要将GET操作稍作修改就可以了。
注意这道题的INSERT操作可能分行输入。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<cstring>
#include<algorithm>
#define F(i,j,n) for(int i=j;i<=n;i++)
#define D(i,j,n) for(int i=j;i>=n;i--)
#define LL long long
#define pa pair<int,int>
#define MAXN 2100000
#define INF 1000000000
#define key c[c[rt][1]][0]
using namespace std;
int pos=1,rt=0,tot=0,n,x,c[MAXN][2],s[MAXN],fa[MAXN];
char v[MAXN],op[10],ch[MAXN];
inline void pushup(int k)
{
if (!k) return;
s[k]=s[c[k][0]]+s[c[k][1]]+1;
}
inline void rotate(int x,int &k)
{
int y=fa[x],z=fa[y],l=(c[y][1]==x),r=l^1;
if (y==k) k=x;
else c[z][c[z][1]==y]=x;
fa[x]=z;fa[y]=x;fa[c[x][r]]=y;
c[y][l]=c[x][r];c[x][r]=y;
pushup(y);pushup(x);
}
inline void splay(int x,int &k)
{
while (x!=k)
{
int y=fa[x],z=fa[y];
if (y!=k)
{
if ((c[y][0]==x)^(c[z][0]==y)) rotate(x,k);
else rotate(y,k);
}
rotate(x,k);
}
}
inline int find(int k,int rank)
{
int l=c[k][0],r=c[k][1];
if (s[l]+1==rank) return k;
else if (s[l]>=rank) return find(l,rank);
else return find(r,rank-s[l]-1);
}
inline void split(int l,int r)
{
int x=find(rt,l-1),y=find(rt,r+1);
splay(x,rt);splay(y,c[rt][1]);
}
inline void newnode(int &k,char ch,int last)
{
k=++tot;
fa[k]=last;
v[k]=ch;
c[k][0]=c[k][1]=0;
s[k]=1;
}
inline void ins(int &k,int l,int r,int last)
{
int mid=(l+r)>>1;
newnode(k,ch[mid],last);
if (l<mid) ins(c[k][0],l,mid-1,k);
if (mid<r) ins(c[k][1],mid+1,r,k);
pushup(k);
}
inline void solveins()
{
scanf("%d%*c",&x);
F(i,1,x)
{
ch[i]=getchar();
while (ch[i]<32||ch[i]>126) ch[i]=getchar();
}
split(pos+1,pos);
ins(key,1,x,c[rt][1]);
pushup(c[rt][1]);pushup(rt);
}
inline void solvedel()
{
scanf("%d%*c",&x);
split(pos+1,pos+x);
fa[key]=0;key=0;
pushup(c[rt][1]);pushup(rt);
}
inline void travel(int k)
{
if (!k) return;
travel(c[k][0]);
printf("%c",v[k]);
travel(c[k][1]);
}
inline void getans()
{
scanf("%d%*c",&x);
split(pos+1,pos+x);
travel(key);printf("\n");
}
int main()
{
newnode(rt,'*',0);newnode(c[rt][1],'*',rt);
pushup(rt);
scanf("%d%*c",&n);
F(i,1,n)
{
scanf("%s%*c",op);
if (op[0]=='M') {scanf("%d%*c",&x);pos=x+1;}
else if (op[0]=='I') solveins();
else if (op[0]=='D') solvedel();
else if (op[0]=='G') getans();
else if (op[0]=='P') pos--;
else pos++;
}
}