块状链表及其应用

思路楼上已经说的很清楚了

看代码注释

代码很丑

#include<cstdio>
#include<cctype>
#include<cstring>
//#define gc() getchar()
#define MAXIN 100000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1024*1024*2+5;
const int MaxSize=/*1700*/4500,MaxNum=/*1700*2*/N*2/MaxSize+100; int num,nxt[MaxNum],sz[MaxNum],pool[MaxNum];
char data[MaxNum][MaxSize],str[N],IN[MAXIN],*SS=IN,*TT=IN; inline int read() {
int now=0,f=1;
register char c=gc();
for(; !isdigit(c); c=gc()) if(c=='-') f=-1;
for(; isdigit(c); now=now*10+c-'0',c=gc());
return now*f;
} inline int New_Block() {
return pool[++num];
}
inline void Del_Block(int v) {
pool[num--]=v;
}
void Init() {
for(int i=1; i<MaxNum; ++i) pool[i]=i;
sz[0]=0, nxt[0]=-1;//新建一个0节点,方便 表头就是0
}
void Merge(int cur,int Nxt) {
memcpy(data[cur]+sz[cur],data[Nxt],sz[Nxt]);
nxt[cur]=nxt[Nxt], sz[cur]+=sz[Nxt];
Del_Block(Nxt);
}
void Maintain() {
for(int cur=0,Nxt=nxt[0]; ~cur; cur=nxt[cur],Nxt=nxt[cur])
while((~Nxt) && sz[cur]+sz[Nxt]<=MaxSize)
Merge(cur,Nxt), Nxt=nxt[cur];//最好不用nxt[Nxt],因为已经合并、删掉了,虽然不影响答案,但还是不该写
}
int Get_Index(int &pos) { //找到pos所在的块,并将pos定位为块内位置
int cur=0;
while((~cur) && pos>sz[cur])//把cur定位到某一块的末尾,不用下一块开头了(pos>=sz)
pos-=sz[cur], cur=nxt[cur];
return cur;
}
void Update(int cur,int Nxt,int len,char *s) { //给新的块Nxt设置数据及指针
nxt[Nxt]=nxt[cur], nxt[cur]=Nxt, sz[Nxt]=len;
memcpy(data[Nxt],s,len);
}
void Split(int cur,int pos) {
if(cur==-1||pos==sz[cur]) return;//不能判!pos!因为后边会直接用nxt[cur],不分裂会跳过该块;而在=sz时不分是没问题的
int Nxt=New_Block();
Update(cur,Nxt,sz[cur]-pos,data[cur]+pos);
sz[cur]=pos;
}
void Insert(int pos,int len) {
int cur=Get_Index(pos),sum=0,Nxt;
Split(cur,pos);
while(sum+MaxSize<=len) {
Nxt=New_Block();
Update(cur,Nxt,MaxSize,str+sum);//先分成尽可能多的整块
sum+=MaxSize, cur=Nxt;
}
if(len-sum)//剩余的单独放到一块
Nxt=New_Block(), Update(cur,Nxt,len-sum,str+sum);
Maintain();
}
void Erase(int pos,int len) {
int cur=Get_Index(pos),Nxt;
Split(cur,pos);
Nxt=nxt[cur];//because of here
while((~Nxt) && len>sz[Nxt])
len-=sz[Nxt], Del_Block(Nxt), Nxt=nxt[Nxt];
Split(Nxt,len);
Del_Block(Nxt), nxt[cur]=nxt[Nxt];
Maintain();
}
void Get_Data(int pos,int len) {
int cur=Get_Index(pos),sum=sz[cur]-pos;
if(len<sum) sum=len;
memcpy(str,data[cur]+pos,sum);
cur=nxt[cur];
while((~cur) && sum+sz[cur]<=len)
memcpy(str+sum,data[cur],sz[cur]), sum+=sz[cur] ,cur=nxt[cur];
if((~cur) && len-sum)
memcpy(str+sum,data[cur],len-sum);
str[len]='\0';
printf("%s\n",str);
}
inline char Get_opt() {
register char c=gc();
while(c!='M'&&c!='I'&&c!='D'&&c!='G'&&c!='P'&&c!='N') c=gc();
return c;
} int main() {
Init();
int t=read(),pos=0,len;
char s[10];
while(t--) {
switch(Get_opt()) {
case 'M':
pos=read();
break;
case 'I':
len=read();
for(int i=0; i<len; ++i) {
str[i]=gc();
if(str[i]<32||str[i]>126) --i;
}
Insert(pos,len);
break;
case 'D':
len=read(),Erase(pos,len);
break;
case 'G':
len=read(),Get_Data(pos,len);
break;
case 'P':
--pos;
break;
case 'N':
++pos;
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. 洛谷 P4008 [NOI2003]文本编辑器

    先推广一下 求赞 我们考虑这样的一个问题 给你一个序列,要求你支持插入,删除,查询单点值 如果用数组,查询O(1),插入删除最坏O(n) 如果用链表,插入删除O(1),查询最坏O(n) 如果用平衡树- ...

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

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

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

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

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

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

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

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

  9. NOI2003 文本编辑器editor

    1507: [NOI2003]Editor Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 1908  Solved: 738[Submit][Statu ...

随机推荐

  1. icns图标的制作

    1. 准备一张无损的png图片(1024x1024) 2. 新建一个文件夹 必须要以iconset为后缀 $ mkdir hgl_pngpic.iconset 3. 使用sips 命令剪切10个不一样 ...

  2. bzoj 2049: [Sdoi2008]Cave 洞穴勘测 (LCT)

    链接:https://www.lydsy.com/JudgeOnline/problem.php?id=2049 题面: 2049: [Sdoi2008]Cave 洞穴勘测 Time Limit: 1 ...

  3. mysql学习基础知识3

    1.视图 简化sql语句的编写,限制可以查看的数据 一张虚拟的表,不占任何内存,查视图时都是临时从所查的表中拿数据 特点: 对于视图的增删改查 都会同步到原始表 对原始表的修改,会同步到视图内可查看的 ...

  4. [GoogleBlog]new-approach-to-china

    https://googleblog.blogspot.com/2010/01/new-approach-to-china.html

  5. python第十四天

    今日内容 1. 带参装饰器  |  wrapper 2. 迭代器 3. 可迭代对象 4.迭代器对象 5.for 迭代器 6.枚举对象 带参装饰器 是指装饰器为被装饰的函数添加新功能,需要外界的参数 - ...

  6. linux date -s

    修改linux的时间可以使用date指令 修改日期: 时间设定成2009年5月10日的命令如下: #date -s 05/10/2009 修改时间: 将系统时间设定成上午10点18分0秒的命令如下.  ...

  7. 关于4A系统(我对4A系统的维护的理解)

    4A系统 4A系统是统一安全管理平台解决方案,指认证Authentication.账号Account.授权Authorization.审计Audit,中文名称为统一安全管理平台解决方案.即将身份认证. ...

  8. java实现八大排序算法

    Arrays.sort() 采用了2种排序算法 -- 基本类型数据使用快速排序法,对象数组使用归并排序. java的Collections.sort算法调用的是归并排序,它是稳定排序 方法一:直接插入 ...

  9. Redis源码 - 事件管理

    Redis 的事件分类 分类 描述 定时器 线程内定时响应,更新缓存时间.关闭非活动的客户端连接等等 pipe 线程间通信,用于其他线程通知主线程退出aeApiPoll() unixsocket 本地 ...

  10. 13、Ajax的使用

    一.AJAX 是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术. a).AJAX = 异步 JavaScript 和 XML. b).AJAX 是一种用于创建快速动态网页的技术. 通过在后 ...