[NOIP模拟]文本编辑器 题解
algorithm
solution
题意描述
九发明了一个完美的文本编辑器。这个编辑器拥有两个光标(cursor),所以九能够同时在两处地方插入和删除文本。这个编辑器除了正常的编辑功能以外,还有一些只有九才知道用处的功能,例如翻转两个光标之间的文本。某一天,九把自己的完美文本编辑器给弄丢了,但是她还有好多好多文本需要处理。于是她想请聪明又智慧的你帮她实现完美文本编辑器的一些功能。
功能列表如下:
功能名称 命令格式 说明
< (move left) < w w 为一个字符,“L”或“R”,表示左光标还是右光标(下同)。该命令将选定光标向左移动,如果已经是最左端则不移动。命令执行成功时输出“T”,若光标已经在最左端,则输出“F”。
> (move right) > w w 同上。与< 命令不同的是,该命令将光标向右移动。命令执行成功时输出“T”,若光标已经在最右端,则输出“F”。
I (insert) I w c 同上。c 是一个可见字符(33≤ ascii 码 ≤ 126),代表在该光标左侧插入该字符。该命令始终输出“T”。
D (delete) D w w 同上。代表删除该光标右侧的一个字符。命令执行成功时输出“T”,若光标右侧没有字符输出“F”。
R (reverse) R 代表翻转左光标和右光标之间的字符。该命令只有左光标在右光标左侧时才能执行。(两光标重合时也不能执行)命令执行成功时输出“T”,否则输“F”。
S (show) S 代表显示当前处理的文本。该命令只输出文本,不输出“T”和“F”。
开始时文本编辑器中有一定内容,左光标在第一个字符左,右光标在最后一个字符右。
注意:在插入和删除操作中,没有被操作的光标与文本的相对左右位置保持不变。特别地,若两个光标重叠,操作后也仍然重叠。
输入格式
第一行是初始时文本编辑器内容。
第二行是一个正整数 N,N 表示操作次数。
接下来有 N 行,每行有一个命令,命令格式如上方表格。
输出格式
对于每个命令,按上方表格要求执行并输出。
样例输入
goodykc
11
I R u
I R l
> L
> L
> L
> L
R
D R
< R
D R
S
样例输出
T
T
T
T
T
T
T
F
T
T
goodluck
样例解释
[goodykc]
[goodykcu]
[goodykcul]
g[oodykcul]
go[odykcul]
goo[dykcul]
good[ykcul]
good[lucky]
good[lucky](光标右边没有字符,失败删除)
good[luck]y
good[luck]
goodluck
数据规模与约定
对于 40% 的数据:1 ≤ N , 初始文本长度 ≤ 100,数据不包含翻转(Reverse)操作;
另有 30% 的数据:1 ≤ N , 初始文本长度 ≤ 105,数据不包含翻转(Reverse)操作;
另有 20% 的数据:1 ≤ N , 初始文本长度 ≤ 105,数据包含翻转(Reverse)操作;
对于 100% 的数据:1 ≤ N , 初始文本长度 ≤ 4 × 106,输出文件大小 ≤ 20MB;
没找到这道题的出处
我不知道为啥有人用splay还有什么双端队列做这道题
反正我考场一看这题就兴奋了
这不直接双向链表暴搞吗
Reverse就直接这样接一下就没了
不过这样将让双向链表无法判断左右方向,那就直接将双向链表变成无向的,指针记录真实的在左边的那个节点的编号,这样就像遍历树那样可以判方向了
考场3kb 70pt,考后+2kb 加上了rev操作
代码权当纪念,诸位看看笑笑就好
反正我是已经看不懂了:p
各种结构体引用三目运算符函数混乱毫无可读性的5kb代码
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<ctime>
#include<cstdlib>
#include<algorithm>
#include<queue>
#include<vector>
#include<map>
using namespace std;
const int INF=999999999;
const int MXN=8E6+5;
struct Node{
int v1,v2;
char c;
}nd[MXN];int pN;
struct Pter{
int id,pre;
/*void Debug(){
cout<<"Pter:"<<id<<"#"<<pre<<endl;
}*/
}p1,p2;
(Pter& A){
Pter Copy=(Pter){A.id,A.pre};
Pter Breturn B;
}
bool IsOK(Pter p){
if(p.id==0||(p.id==INF&&p.pre==INF)) return 0;
return 1;
}
char str[MXN];int rpL,rpR;
void Build(){
int sLen=strlen(str);pN=sLen;
for(int i=1;i<sLen;i++) nd[i].v2=i+1;
for(int i=2;i<=sLen;i++) nd[i].v1=i-1;
[1].v1=0;nd[sLen].v2=INF;
ndfor(int i=1;i<=sLen;i++) nd[i].c=str[i-1];
[0].v1=-INF,nd[0].v2=1;
nd.id=1;p1.pre=0;
p1.id=INF;p2.pre=sLen;
p2
=1;rpR=strlen(str)+1;
rpL}
(Pter& nw){
Pter Leftint le=nw.pre;
if(le==0) return (Pter){0,0};
int lele=(nd[le].v1!=nw.id)?nd[le].v1:nd[le].v2;
=(Pter){le,lele};
Pter preturn p;
}
(Pter& nw){
Pter Rightint id=nw.id;if(id==INF) return (Pter){INF,INF};
int ri=(nd[id].v1==nw.pre)?nd[id].v2:nd[id].v1;
=(Pter){ri,id};
Pter preturn p;
}
void Ins(Pter& nw,char c){
++;nd[pN].c=c;
pNint id=nw.id;
=Left(nw);
Pter le[pN].v1=le.id;nd[pN].v2=nw.id;
nd
int& leR=(nd[le.id].v1==id?nd[le.id].v1:nd[le.id].v2);
=pN;
leR
if(IsOK(Right(nw))){
int& nwL=(nd[id].v1==nw.pre?nd[id].v1:nd[id].v2);
=pN;
nwL}
.pre=pN;
nw}
bool Del(Pter& nw){
if(nw.id==INF){printf("F");return 0;}
=Left(nw);
Pter le=Right(nw);
Pter ri
int& leR=(nd[le.id].v1==nw.id?nd[le.id].v1:nd[le.id].v2);
=ri.id;
leR
if(IsOK(Right(ri))){
int& riL=(nd[ri.id].v1==nw.id?nd[ri.id].v1:nd[ri.id].v2);
=le.id;
riL}
.id=ri.id;
nw.pre=le.id;
nw("T");
printfreturn 1;
}
void Rev(){
=Copy(p1);
Pter L=Left(L);
Pter LL=Copy(p2);
Pter RR=Left(RR);
Pter R//cout<<L.id<<","<<LL.id<<"#"<<R.id<<","<<RR.id<<"###";
int& leleR=(nd[LL.id].v1==L.id?nd[LL.id].v1:nd[LL.id].v2);
=R.id;
leleR
int& leL =(nd[L.id].v1==LL.id?nd[L.id].v1:nd[L.id].v2);
=RR.id;
leL
if(IsOK(Right(RR))){
int& ririL=(nd[RR.id].v1==R.id?nd[RR.id].v1:nd[RR.id].v2);
=L.id;
ririL}
int& riR =(nd[R.id].v1==RR.id?nd[R.id].v1:nd[R.id].v2);
=LL.id;
riR
.id=R.id;
p1.pre=L.id;
p2}
void Print(){
=(Pter){nd[0].v2,0};
Pter pwhile(IsOK(Right(p))){//p.Debug();
("%c",nd[p.id].c);
printf=Right(p);
p
}
}
int main(){
("%s",str);
scanf();
Buildint qN;cin>>qN;
for(int i=1;i<=qN;i++){
char typ[2];scanf("%s",typ);
char wch[2],ccc[2];
switch(typ[0]){
case '<':{//cout<<"#";
("%s",wch);Pter& p=(wch[0]=='L')?p1:p2;
scanf=Left(p);
Pter lpif(!IsOK(lp)) printf("F");
else{
("T");
printf=lp;
p((wch[0]=='L')?rpL:rpR) --;
}
//p.Debug();
break;
}case '>':{//cout<<"V";
("%s",wch);Pter& p=(wch[0]=='L')?p1:p2;
scanf=Right(p);
Pter rpif(!IsOK(rp)) printf("F");
else{
("T");
printf=rp;
p((wch[0]=='L')?rpL:rpR) ++;
}
//p.Debug();
break;
}case 'I':{
("%s",wch);Pter& p=(wch[0]=='L')?p1:p2;
scanf& otr=(wch[0]=='L')?p2:p1;
Pterint& ooo=(wch[0]=='L')?rpL:rpR;
int& ppp=(wch[0]=='L')?rpR:rpL;
("%s",ccc);char c=ccc[0];
scanfint bomb=0;
if(p.id==otr.id&&p.pre==otr.pre) bomb=1;
(p,c);
Ins("T");
printfif(bomb==1) otr.id=p.id,otr.pre=p.pre;
if(ooo<=ppp) ppp++;
++;
ooobreak;
}case 'D':{
("%s",wch);Pter& p=(wch[0]=='L')?p1:p2;
scanf& otr=(wch[0]=='L')?p2:p1;
Pterint& ooo=(wch[0]=='L')?rpL:rpR;
int& ppp=(wch[0]=='L')?rpR:rpL;
int bomb=0;
if(p.id==otr.id&&p.pre==otr.pre) bomb=1;
else if(otr.pre==p.id) bomb=2;
bool isSuc=Del(p);
if(bomb==1) otr.id=p.id,otr.pre=p.pre;
else if(bomb==2) otr.pre=p.pre;
//p.Debug();otr.Debug();
if(isSuc){
if(ooo<ppp) ppp--;
}
break;
}case 'R':{
//printf("F");
if(rpL<rpR){
();
Rev("T");
printf}else{
("F");
printf}
break;
}case 'S':{
();
Printbreak;
}
}
//cout<<p1.id<<','<<p1.pre<<"|"<<p2.id<<","<<p2.pre<<endl;
//cout<<"!"<<rpL<<','<<rpR;
("\n");
printf}
return 0;
}
/*
ab
111
R
S
goodykc
11
I R u
I R l
> L
> L
> L
> L
R
D R
< R
D R
S
*/