块状链表及其应用

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

看代码注释

代码很丑

#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. 「Manacher算法」学习笔记

    觉得这篇文章写得特别劲,插图非常便于理解. 目的:求字符串中的最长回文子串. 算法思想 考虑维护一个数组$r[i]$代表回文半径.回文半径的定义为:对于一个以$i$为回文中心的奇数回文子串,设其为闭区 ...

  2. [BJOI2019] 删数

    https://www.luogu.org/problemnew/show/P5324 题解 首先我们需要弄清这个答案是什么. 对于一个长度为n的序列,那么它先删的肯定是\(n\),删完之后它就会跳到 ...

  3. php 两个数组,若键相同,则值合并

    <?php $arr1 = array('9' => '4.08', '10' => '0.10', '11' => '4.08', '12' => '0.01'); $ ...

  4. LOJ#2087 国王饮水记

    解:这个题一脸不可做... 比1小的怎么办啊,好像没用,扔了吧. 先看部分分,n = 2简单,我会分类讨论!n = 4简单,我会搜索!n = 10,我会剪枝! k = 1怎么办,好像选的那些越大越好啊 ...

  5. 51nod1556 计算(默慈金数)

    Problem 有一个\(1*n\)的矩阵,固定第一个数为\(1\),其他填正整数, 且相邻数的差不能超过\(1\),求方案数. \(n\le 10^6\) Solution 容易发现答案是\(f_n ...

  6. 三步法搞定CTF中的SQL注入题型

    三步法: 一.找到注入点 二.Fuzz出未过滤字符 三.构造payload/写脚本 例题1 打开题目: 第一步,寻找注入点. 输入用户名123456,密码123456,返回结果username err ...

  7. Eclipse——手把手教新手安装Eclipse

    一.准备工作:安装JRE和JDK. 全名分别为:Java Runtime Environmen和Java SE Development Kit,推荐直接在某度软件中心下载即可,注意区分64位和32位. ...

  8. git bash 支持中文

      1. 编辑etc\gitconfig文件,在文件末尾增加以下内容: [gui] encoding = utf-8 #代码库统一使用utf-8 pathnameencoding = utf-8 #支 ...

  9. CSS white-space属性详解

    概述 CSS的white-space属性用于指定如何处理容器中的空白字符,例如:空格( ).换行(\n).缩进(\t)等. white-space出自CSS1,适用于块状元素,具有继承性,支持IE 5 ...

  10. ue4 staticMesh属性记录

    Light Map Resolution 光照贴图分辨率 Generate Mesh Distancde Field 生成网格距离场(一种不怎么消耗性能的阴影) http://api.unrealen ...