洛谷.4008.[NOI2003]editor文本编辑器(块状链表)
st(n)表示sqrt(n)
为使块状链表不会退化,通常将每块的大小S维持在[st(n)/2,2st(n)]中,这样块数C也一定[st(n)/2,2st(n)]中
在此使用另一种方法(方便)维持这种性质:保证任意相邻两块的size相加>st(n),并且每块大小<=st(n)
这样可保证块数C在[st(n),2st(n)]中
在每次块状链表结构改变后执行maintain,顺序扫描链表,将相邻两块能合并就合并
1.注意Insert的读入,会有回车
2.为避免各种翻车,还是加上判断!=-1好
3.可以根据操作分布,适当调大块的大小
//72ms 4.32MB(轻松虐掉Rank2)
#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()
{
#ifndef ONLINE_JUDGE
freopen("4008.in","r",stdin);
#endif
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;
}
洛谷.4008.[NOI2003]editor文本编辑器(块状链表)的更多相关文章
- 洛谷 P4008 [NOI2003]文本编辑器 解题报告
P4008 [NOI2003]文本编辑器 题目描述 很久很久以前,\(DOS3.x\)的程序员们开始对 \(EDLIN\) 感到厌倦.于是,人们开始纷纷改用自己写的文本编辑器⋯⋯ 多年之后,出于偶然的 ...
- 洛谷 P4008 [NOI2003]文本编辑器
先推广一下 求赞 我们考虑这样的一个问题 给你一个序列,要求你支持插入,删除,查询单点值 如果用数组,查询O(1),插入删除最坏O(n) 如果用链表,插入删除O(1),查询最坏O(n) 如果用平衡树- ...
- [洛谷日报#204] StackEdit——Markdown 编辑器的功能介绍
本文同时发表于洛谷日报,您也可以通过洛谷博客进行查看. 1.介绍与开始使用 1.1 这是什么? StackEdit是基于PageDown.Stack Overflow和其他堆栈交换站点使用的Markd ...
- [NOI2003][bzoj1507] 文本编辑器 editor [splay]
其实看明白了就是一道水题 毕竟模板 splay敲一发,插入一个串的时候先把它构建成一棵平衡树,再挂到原来的splay上面去即可 没别的了,上代码 #include<iostream> #i ...
- 【洛谷 P4052】 [JSOI2007]文本生成器(AC自动机,DP)
题目链接 AC自动机上dp第一题嗷. 如果直接求可读文本的数量,显然要容斥,不好搞. 于是考虑求不可读文本的数量,再用\(26^m\)减去其即可. 建出AC自动机,如果一个节点为单词结尾或其fail链 ...
- 洛谷 P4408 [NOI2003]逃学的小孩
题目传送门 题目描述 Chris家的电话铃响起了,里面传出了Chris的老师焦急的声音:“喂,是Chris的家长吗?你们的孩子又没来上课,不想参加考试了吗?”一听说要考试,Chris的父母就心急如焚, ...
- 洛谷【P2201】数列编辑器
我对模拟的理解:http://www.cnblogs.com/AKMer/p/9064018.html 题目传送门:https://www.luogu.org/problemnew/show/P220 ...
- 洛谷 P4408 [NOI2003] 逃学的小孩 题解
Analysis 题意虽然说先去谁家再去谁家,但是我们不需要管这个,因为AA.BB.CC三个点我们可以任意互相交换它们所代表的对象,所以题目要求的就是在一棵树上找到3个点AA.BB.CC令AB+BCA ...
- Ubuntu Text editor文本编辑器相关设置
刚开始不熟悉Ubuntu,设置个文本编辑界面都难找到: 打开后在顶上的导航栏,下拉框内有preferences: 里面可以设置视图.字体颜色等
随机推荐
- ubuntu下好用的音乐播放器audacious
audacious是ubuntu下一款非常好用的音乐播放器,万能的音乐播放器而且简洁美观,可以播放ape各种无损发烧音乐格式. 如果想听音乐的话,现在百度音乐,酷我音乐,酷狗音乐等都是有网络播放器的, ...
- mysql授权报错 ERROR 1819 (HY000): Your password does not satisfy the current policy requirements
授权用户时报错,ERROR 1819 (HY000): Your password does not satisfy the current policy requirements 原因为其实与val ...
- mysql系列九、mysql语句执行过程及运行原理(分组查询和关联查询原理)
一.背景介绍 了解一个sql语句的执行过程,了解一部分都做了什么,更有利于对sql进行优化,因为你知道它的每一个连接.where.分组.子查询是怎么运行的,都干了什么,才会知道怎么写是不合理的. 大致 ...
- Websphere MQ Cluster
大纲: 1.什么是集群 2.建立一个基本的集群 3.DISPLAY命令 4.负载均衡 5.高级配置和管理 6.答疑 7.关于文章.红宝书等 一. 什么是集群 集群就是Websphere M ...
- windows下设置计划任务自动执行PHP脚本
背景: 环境部署在linux下或者windows中,可以使用windows的自动任务设置自动执行脚本执行一些日常运维任务 图形界面设置相对比较简单 准备工作: wamp(集成的PHP执行环境) 已经写 ...
- npm install 报错(npm ERR! errno -4048,Error: EPERM: operation not permitted,)解决方法
npm ERR! path E:\SouthernPowerGridProject\web_project\AutoOPS\autoops\node_modules\fsevents\node_mod ...
- 分享我对JS插件开发的一些感想和心得
本文阅读目录: •起因•如何开发一个轻量级的适用性强的插件•总结 起因 如果大家平时做过一些前端开发方面的工作,一定会有这样的体会:页面需要某种效果或者插件的时候,我们一般会有两种选择: 1.上网查找 ...
- MySQL表的定期分析检查优化
Analyze Table 分析表 MySQL 的Optimizer(优化元件)在优化SQL语句时,首先需要收集一些相关信息,其中就包括表的cardinality(可以翻译为“散列程度”),它表示 ...
- 【ES】学习3-请求体查询
1.空查询 GET /index_2014*/type1,type2/_search {} GET /_search { , } 2.查询表达式 DSL只需将查询语句传递给 query 参数 GET ...
- RzPageControl Tab拖拽 移动