先推广一下

求赞

我们考虑这样的一个问题

给你一个序列,要求你支持插入,删除,查询单点值

如果用数组,查询O(1),插入删除最坏O(n)

如果用链表,插入删除O(1),查询最坏O(n)

如果用平衡树……

不要跟我说平衡树

那么我们是否可以考虑:将一个一个的数组以链表的形式串起来,这样是否会提高操作的效率,又是否会降低一些操作的效率呢?

可以手动模拟一下各种操作

块状链表就是这样一个略显暴力的算法

但其复杂度较为优秀,所以在很多地方的应用都非常广

用一句话说叫“弱弱联合”

码量稍大,但极易理解,打着打着就打出两百K行

先介绍一下比较基本的操作吧

Spilt

当一个块的长度过大,我们就可以考虑将其分裂成两个较小的块。

在处理类似于插入或者删除这类操作时,我们可以先从当前位置将其分裂成两个块,这样就可以十分方便的进行操作了。

Merge

同理,就是\(Split\)的逆运算。

部分Maintain

姑且称其为部分maintain吧,我也不知道叫什么。

在进行操作时,我们可能会使得一些块过大,一些块过小。

所以我们需要通过\(Spilt\)或者\(Merge\)来调整。

我们发现,在进行操作时所需要考虑的需要维护的块:区间前的那一块与区间开头块;区间末尾块与区间后的那一块。

这样做可能会使得块状链表没有在经过完整maintain操作时平衡,但会大大减少维护时的常数,而平衡程度也可以接受。

一般采用的维护方法:保证相邻两块大小加起来大于\(\sqrt{n}\),但每块大小不超过\(\sqrt{n}\),这样可以较好的维护平衡,同时不用考虑当块较大时的\(Split\)操作。

这是作者经过权衡后得出的做法,实测复杂度优秀,复杂度为\(O(1)\)。


然后,我们切入正题。

Insert

查找光标块内的位置,在此位置将块分裂,然后将字符串一块一块地插入

Delete

同理

Get

不需要分裂,直接利用\(memcpy\)函数,对其进行复制粘贴即可

代码中有较详细注释,贴代码

#include<bits/stdc++.h>
using namespace std;
char xch,xB[1<<15],*xS=xB,*xTT=xB;
#define getc() (xS==xTT&&(xTT=(xS=xB)+fread(xB,1,1<<15,stdin),xS==xTT)?0:*xS++)
inline int read()
{
int x=0,f=1;char ch=getc();
while(ch<'0'|ch>'9'){if(ch=='-')f=-1;ch=getc();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getc();}
return x*f;
}//为了使程序跑得更快所使用的读入优化
const int maxn=2e3+10;
struct node{
int nex,siz;//每一块数组的后继以及大小
char a[maxn<<1];
}b[maxn<<2];
int pool[maxn<<2],cnt,curpos;//内存池、指针以及当前光标位置
inline int modi(){return pool[cnt++];}//内存分配
inline void dele(int x){pool[--cnt]=x;}//内存回收
inline void init()
{
for(int i=1;i<(maxn<<2);++i) pool[i]=i;//维护内存池,动态分配回收内存
cnt=1;
b[0].siz=0,b[0].nex=-1;//新建一个0号节点,方便操作
}
inline void add(int x,int y,int num,char c[])//在第x块后添加一个编号为y的块,长度为num
{
if(y!=-1)
{
b[y].nex=b[x].nex,b[y].siz=num;
memcpy(b[y].a,c,num);
}
b[x].nex=y;
}
inline void merge(int x,int y)//将第x块和第y块合并
{
memcpy(b[x].a+b[x].siz,b[y].a,b[y].siz);
b[x].siz+=b[y].siz,b[x].nex=b[y].nex;
dele(y);
}
inline void split(int cur,int pos)//将第cur块从pos处分割
{
if(cur==-1||pos==b[cur].siz) return ;
add(cur,modi(),b[cur].siz-pos,b[cur].a+pos);
b[cur].siz=pos;
}
inline int pos(int &x)//寻找当前光标所在的块和块内位置
{
int now=0;
while(now!=-1&&x>b[now].siz) x-=b[now].siz,now=b[now].nex;
return now;
}
inline void insert(int p,int num,char c[])//在p位置之后插入长度为num的字符串
{
int now=pos(p);
split(now,p);
int tot=0,nb,st=now;
while(tot+maxn<=num)//维护块状链表平衡
{
nb=modi();
add(now,nb,maxn,c+tot);
tot+=maxn;
now=nb;
}
if(num-tot)
nb=modi(),add(now,nb,num-tot,c+tot);
if(b[now].siz+b[nb].siz<maxn&&nb!=-1)//不用对整个链表进行判断,部分maintain
merge(now,nb),nb=b[now].nex;
if(b[st].siz+b[b[st].nex].siz<maxn&&b[st].nex!=-1)//同理
merge(st,b[st].nex);
// maintain();
}
inline void erase(int p,int num)//在p位置之后删除长度为num的字符串
{
int now=pos(p);
split(now,p);
int nex=b[now].nex;
while(nex!=-1&&num>b[nex].siz)
num-=b[nex].siz,nex=b[nex].nex;
split(nex,num);
nex=b[nex].nex;
for(int i=b[now].nex;i!=nex;i=b[now].nex)
b[now].nex=b[i].nex,dele(i);
while(b[now].siz+b[nex].siz<maxn&&nex!=-1)//不用对整个链表进行判断,部分maintain
merge(now,nex),nex=b[now].nex;
// maintain();
}
char ans[20000000];
inline void get(int p,int num)//输出p位置后长度为num的字符串
{
int cur=pos(p);
int tot=b[cur].siz-p;
if(num<tot) tot=num;
memcpy(ans,b[cur].a+p,tot);
int now=b[cur].nex;
while(now!=-1&&num>=tot+b[now].siz)
{
memcpy(ans+tot,b[now].a,b[now].siz);
tot+=b[now].siz,now=b[now].nex;
}
if(num-tot>0&&now!=-1)
memcpy(ans+tot,b[now].a,num-tot);
ans[num]='\0';//为了不清空,用\0结束
printf("%s\n",ans);
}
inline char opt()
{
char c=getc();
while(c!='M'&&c!='I'&&c!='D'&&c!='G'&&c!='P'&&c!='N') c=getc();
return c;
}//为了不与读入优化冲突
int main()
{
// freopen("3.in","r",stdin);
// freopen("3.ans","w",stdout);
init();
int m; scanf("%d",&m);
for(int i=1;i<=m;++i)
{
switch(opt())
{
case 'M':curpos=read();break;
case 'I':
int tmp;
tmp=read();
for(int i=0;i<tmp;++i)
{
ans[i]=getc();
if(ans[i]<32||ans[i]>126) --i;
}
insert(curpos,tmp,ans);
break;
case 'D':
tmp=read();
erase(curpos,tmp);
break;
case 'G':
tmp=read();
get(curpos,tmp);
break;
case 'P':--curpos;break;
case 'N':++curpos;break;
}
}
return 0;
}

洛谷 P4008 [NOI2003]文本编辑器的更多相关文章

  1. 洛谷 P4008 [NOI2003]文本编辑器 解题报告

    P4008 [NOI2003]文本编辑器 题目描述 很久很久以前,\(DOS3.x\)的程序员们开始对 \(EDLIN\) 感到厌倦.于是,人们开始纷纷改用自己写的文本编辑器⋯⋯ 多年之后,出于偶然的 ...

  2. luogu P4008 [NOI2003]文本编辑器 splay 块状链表

    LINK:文本编辑器 这个东西感觉块状链表写细节挺多 (块状链表本来就难写 解释一下块状链表的做法:其实是一个个数组块 然后利用链表给链接起来 每个块的大小为sqrt(n). 这样插入删除的时候直接暴 ...

  3. P4008 [NOI2003]文本编辑器

    思路 FHQ Treap的板子 用FHQ Treap维护中序遍历序列即可 然后数组开够! 代码 #include <cstdio> #include <cstring> #in ...

  4. [NOI2003] 文本编辑器 (splay)

    复制炸格式了,就不贴题面了 [NOI2003] 文本编辑器 Solution 对于光标的移动,我们只要记录一下现在在哪里就可以了 Insert操作:手动维护中序遍历结果,即每次取中点像线段树一样一样递 ...

  5. [NOI2003]文本编辑器 [Fhq Treap]

    [NOI2003]文本编辑器 没啥好说的 就是个板子 #include <bits/stdc++.h> // #define int long long #define rep(a , b ...

  6. cogs 330. [NOI2003] 文本编辑器

    ★★★   输入文件:editor2003.in   输出文件:editor2003.out   简单对比 时间限制:2 s   内存限制:128 MB [问题描述] 很久很久以前,DOS3.x的程序 ...

  7. 【洛谷 P4008】 [NOI2003]文本编辑器 (Splay)

    题目链接 \(Splay\)先练到这吧(好像还有道毒瘤的维护数列诶,算了吧) 记录下光标的编号,维护就是\(Splay\)基操了. 另外数据有坑,数据是\(Windows\)下生成了,回车是'\n\r ...

  8. 洛谷 - P4008 - 文本编辑器 - 无旋Treap

    https://www.luogu.org/problem/P4008 无旋Treap也可以维护序列. 千万要注意要先判断p节点存在才进行Show操作,不然输出一个'\0'(或者RecBin里面的东西 ...

  9. 洛谷.4008.[NOI2003]editor文本编辑器(块状链表)

    题目链接 st(n)表示sqrt(n) 为使块状链表不会退化,通常将每块的大小S维持在[st(n)/2,2st(n)]中,这样块数C也一定[st(n)/2,2st(n)]中 在此使用另一种方法(方便) ...

随机推荐

  1. skynet debug console 使用

    预读 关于如何使用 skynet 可以参考 wiki 文档 更多实战内容见 https://www.lanqiao.cn/courses/2770 优惠码:2CZ2UA5u 环境测试搭建 使用示例代码 ...

  2. 初遇SpringBoot踩坑与加载静态文件遇到的坑

                              SpringBoot开发 创建SpringBoot项目 大家都知道SpringBoot开发非常快,创建SpringBoot项目时,勾上SpringW ...

  3. Docker学习(11) Dockerfile指令

    Dockerfile指令 指令格式 FROM MAINTAINER RUN EXPOSE CMD ENTRYPOINT ADD COPY VOLUME WORKDIR ENV USER ONBUILD ...

  4. zk客户端及锁的使用

    1.生成zk客户端对象 private CuratorFramework buildClient() { logger.info("zookeeper registry center ini ...

  5. 降维-基于RDD的API

    降维-基于RDD的API Singular value decomposition (SVD) Performance SVD Example Principal component analysis ...

  6. PyTorch全连接ReLU网络

    PyTorch全连接ReLU网络 1.PyTorch的核心是两个主要特征: 一个n维张量,类似于numpy,但可以在GPU上运行 搭建和训练神经网络时的自动微分/求导机制 本文将使用全连接的ReLU网 ...

  7. 快手推荐系统及 Redis 升级存储

    快手推荐系统及 Redis 升级存储  借傲腾 补上 DRAM 短板 内容简介: 作为短视频领域的领先企业,快手需要不断导入更先进的技术手段来调整和优化其系统架构,以应对用户量和短视频作品数量的爆炸式 ...

  8. 如果你这么去理解HashMap就会发现它真的很简单

    Java中的HashMap相信大家都不陌生,也是大家编程时最常用的数据结构之一,各种面试题更是恨不得掘地三尺的去问HashMap.HashTable.ConcurrentHashMap,无论面试题多么 ...

  9. 详解详解Java中static关键字和final关键字的功能

    摘要:static关键字和final关键字是Java语言的核心,深入理解他们的功能非常重要. 本文分享自华为云社区<Java: static关键字与final关键字>,原文作者:唐里 . ...

  10. CVPR2021|一个高效的金字塔切分注意力模块PSA

    ​ 前言: 前面分享了一篇<继SE,CBAM后的一种新的注意力机制Coordinate Attention>,其出发点在于SE只引入了通道注意力,CBAM的空间注意力只考虑了局部区域的信息 ...