2016-06-18 当时关于块状链表的想法是错误的,之前维护的是一个动态的$\sqrt{n}$,所以常数巨大,今天才知道原因TwT,请不要参照这个程序为模板!!!

模板题水啊水~~~

第一次写块状链表,先写一个模板题(⊙o⊙)

块状链表虽然效率大概在O(n√n),但它几乎没有常数,相比较理论上复杂度较快的O(nlogn)的Splay还是要快,因为Splay常数太大啦!感觉比较笨重。也许这就是许多OIer热衷分块的原因吧。

BZOJ 1507(因为O2优化,没有内存池)

#include<cmath>
#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define for1(i,a,n) for(int i=(a);i<=(n);++i)
#define for2(i,a,n) for(int i=(a);i<(n);++i)
#define for3(i,a,n) for(int i=(a);i>=(n);--i)
#define for4(i,a,n) for(int i=(a);i>(n);--i)
#define CC(i,a) memset(i,a,sizeof(i))
using namespace std;
int now=0,min_tot=0,max_tot=0,tot=0,sq;
struct BLOCK{
BLOCK();
BLOCK *next;
char c[100003];
int size;
}*head,*null;
BLOCK::BLOCK(){next=null;size=0;}
inline void build(){
null=new BLOCK;
*null=BLOCK();
head=new BLOCK;
}
inline BLOCK *find(int &k){
BLOCK *r=head;
while (k-r->size>0&&r->next!=null) k-=r->size,r=r->next;
return r;
}
inline void cut(BLOCK *r,int pos){
BLOCK *k=new BLOCK;
k->size=r->size-pos;
for1(i,pos+1,r->size) k->c[i-pos]=r->c[i];
r->size=pos;
k->next=r->next;
r->next=k;
}
inline void merge(BLOCK *r){
if (r->next==null) return;
BLOCK *k=r->next;
for1(i,1,k->size) r->c[r->size+i]=k->c[i];
r->size+=k->size;
r->next=k->next;
delete k;
}
inline void balance(){
BLOCK *r=head;
sq=floor(sqrt(tot)); min_tot=sq/2; max_tot=sq*2;
while (r!=null){
if (r->size<min_tot) if (r->next==null) return; else {merge(r); continue;}
else if (r->size>max_tot) {cut(r,r->size/2); continue;}
r=r->next;
}
}
inline void work1(int x){
int pnow=now; BLOCK *r=find(pnow),*k=new BLOCK,*sta=k;
cut(r,pnow);
for1(i,1,x){
char cs=getchar();
while (cs=='\n') cs=getchar();
k->size++; k->c[k->size]=cs;
if (k->size>=sq){
k->next=new BLOCK;
k=k->next;
}
}k->next=r->next;r->next=sta;
}
inline void work2(int x){
int pnow=now; BLOCK *r=find(pnow);
if (x<=r->size-pnow){
cut(r,pnow); cut(r->next,x);
BLOCK *xx=r->next;
r->next=r->next->next;
delete xx;
return;
}
x-=r->size-pnow;
cut(r,pnow);
BLOCK *xy=r->next;
r->next=r->next->next;
delete xy;
while (x-r->next->size>0&&r->next!=null){
x-=r->next->size;
BLOCK *xx=r->next;
r->next=r->next->next;
delete xx;
}cut(r->next,x);
BLOCK *xx=r->next;
r->next=r->next->next;
delete xx;
}
inline void work3(int x){
int pnow=now; BLOCK *r=find(pnow);
if (x<=r->size-pnow){for1(i,pnow+1,pnow+x) putchar(r->c[i]); return;}
for1(i,pnow+1,r->size) putchar(r->c[i]);
x-=r->size-pnow; r=r->next;
while (x-r->size>0&&r!=null){
for1(i,1,r->size) putchar(r->c[i]);
x-=r->size; r=r->next;
}for1(i,1,x)putchar(r->c[i]);
}
int main(){
build();
int n;scanf("%d\n",&n);
while (n--){
char str=getchar(),st=getchar();
int x;
switch(str){
case 'I':
while (st!=' ') st=getchar();
scanf("%d",&x); tot+=x; sq=floor(sqrt(tot)); min_tot=sq/2; max_tot=sq*2;
work1(x);
balance();
scanf("\n");
break;
case 'M':
while (st!=' ') st=getchar();
scanf("%d\n",&now);
break;
case 'D':
while (st!=' ') st=getchar();
scanf("%d\n",&x); work2(x);
tot-=x; balance();
break;
case 'G':
while (st!=' ') st=getchar();
scanf("%d\n",&x); work3(x); printf("\n");
break;
case 'P':
now--;st=getchar();st=getchar();st=getchar();
break;
case 'N':
now++;st=getchar();st=getchar();st=getchar();
break;
}
}
return 0;
}

Tyvj P2388(Windows下评测的new和delete慢得飞起,所以写了个内存池)

#include<cmath>
#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define for1(i,a,n) for(int i=(a);i<=(n);++i)
#define for2(i,a,n) for(int i=(a);i<(n);++i)
#define for3(i,a,n) for(int i=(a);i>=(n);--i)
#define for4(i,a,n) for(int i=(a);i>(n);--i)
#define CC(i,a) memset(i,a,sizeof(i))
using namespace std;
int now=0,min_tot=0,max_tot=0,tot=0,sq;
struct BLOCK{
BLOCK();
BLOCK *next;
char c[10003];
int size;
}*head,*null,pool[10003];
int top=0;
BLOCK::BLOCK(){next=null;size=0;}
inline BLOCK *newBLOCK(){
BLOCK *t=&pool[top++];
t->next=null; t->size=0;
return t;
}
inline void build(){
null=newBLOCK();
null->next=null; null->size=0;
head=newBLOCK();
}
inline BLOCK *find(int &k){
BLOCK *r=head;
while (k-r->size>0&&r->next!=null) k-=r->size,r=r->next;
return r;
}
inline void cut(BLOCK *r,int pos){
BLOCK *k=newBLOCK();
k->size=r->size-pos;
for1(i,pos+1,r->size) k->c[i-pos]=r->c[i];
r->size=pos;
k->next=r->next;
r->next=k;
}
inline void merge(BLOCK *r){
if (r->next==null) return;
BLOCK *k=r->next;
for1(i,1,k->size) r->c[r->size+i]=k->c[i];
r->size+=k->size;
r->next=k->next;
//delete k;
}
inline void balance(){
BLOCK *r=head;
sq=floor(sqrt(tot)); min_tot=sq/2; max_tot=sq*2;
while (r!=null){
if (r->size<min_tot) if (r->next==null) return; else {merge(r); continue;}
else if (r->size>max_tot) {cut(r,r->size/2); continue;}
r=r->next;
}
}
inline void work1(int x){
int pnow=now; BLOCK *r=find(pnow),*k=newBLOCK(),*sta=k;
cut(r,pnow);
for1(i,1,x){
char cs=getchar();
while (cs=='\n') cs=getchar();
k->size++; k->c[k->size]=cs;
if (k->size>=sq){
k->next=newBLOCK();
k=k->next;
}
}k->next=r->next;r->next=sta;
}
inline void work2(int x){
int pnow=now; BLOCK *r=find(pnow);
if (x<=r->size-pnow){
cut(r,pnow); cut(r->next,x);
BLOCK *xx=r->next;
r->next=r->next->next;
//delete xx;
return;
}
x-=r->size-pnow;
cut(r,pnow);
BLOCK *xy=r->next;
r->next=r->next->next;
//delete xy;
while (x-r->next->size>0&&r->next!=null){
x-=r->next->size;
BLOCK *xx=r->next;
r->next=r->next->next;
//delete xx;
}cut(r->next,x);
BLOCK *xx=r->next;
r->next=r->next->next;
//delete xx;
}
inline void work3(int x){
int pnow=now; BLOCK *r=find(pnow);
if (x<=r->size-pnow){for1(i,pnow+1,pnow+x) putchar(r->c[i]); return;}
for1(i,pnow+1,r->size) putchar(r->c[i]);
x-=r->size-pnow; r=r->next;
while (x-r->size>0&&r!=null){
for1(i,1,r->size) putchar(r->c[i]);
x-=r->size; r=r->next;
}for1(i,1,x)putchar(r->c[i]);
}
int main(){
build();
int n;scanf("%d\n",&n);
while (n--){
char str=getchar(),st=getchar();
int x;
switch(str){
case 'I':
while (st!=' ') st=getchar();
scanf("%d",&x); tot+=x; sq=floor(sqrt(tot)); min_tot=sq/2; max_tot=sq*2;
work1(x);
balance();
scanf("\n");
break;
case 'M':
while (st!=' ') st=getchar();
scanf("%d\n",&now);
break;
case 'D':
while (st!=' ') st=getchar();
scanf("%d\n",&x); work2(x);
tot-=x; balance();
break;
case 'G':
while (st!=' ') st=getchar();
scanf("%d\n",&x); work3(x); printf("\n");
break;
case 'P':
now--;st=getchar();st=getchar();st=getchar();
break;
case 'N':
now++;st=getchar();st=getchar();st=getchar();
break;
}
}
return 0;
}

分块真心强大,,,

【BZOJ 1507】【NOI 2003】&【Tyvj P2388】Editor 块状链表模板题的更多相关文章

  1. 【BZOJ-1507】Editor 块状链表

    1507: [NOI2003]Editor Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 3397  Solved: 1360[Submit][Stat ...

  2. BZOJ 4864: [BeiJing 2017 Wc]神秘物质 (块状链表/平衡树 )

    这就是一道数据结构裸题啊,最大极差就是区间最大值减最小值,最小极差就是相邻两个数差的最小值.然后平衡树splay/treap或者块状链表维护就行了. 第一次自己写块状链表,蛮好写,就是长..然后就BZ ...

  3. 【块状链表】AutSky_JadeK的块状链表模板+总结(STL版)

    Part 1.块状链表.   定位 插入 删除 数组 O(1) O(n) O(n) 链表 O(n) O(1) O(1) 对于线性表的以上常见操作来说,数组和链表都无法有效地解决.但是,若我们将链表的每 ...

  4. bzoj 4622: [NOI 2003] 智破连环阵【dfs+匈牙利算法】

    一个炸弹炸一个区间的武器,想到二分图匹配 但是直接dfs断点显然不行,预处理出dis[i]为i到m的至多值来最优性剪枝,并且标记ok[i][j]为炸弹i可以炸到j武器,mx[i][j]为i炸弹从j武器 ...

  5. BZOJ 1509[NOI 2003]逃学的小孩 树形dp

    1509: [NOI2003]逃学的小孩 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 995  Solved: 505[Submit][Status][ ...

  6. 【BZOJ 3188】【Coci 2011】Upit Splay模板题

    转啊转终于转出来了,然而我的模板跟陈竞潇学长的模板一模一样,还是太弱啊,第一次用指针. #include<cstdio> #include<cstring> #include& ...

  7. BZOJ 1036 树的统计Count 树链剖分模板题

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...

  8. [BZOJ 2006] [NOI 2010]超级钢琴(贪心+ST表+堆)

    [BZOJ 2006] [NOI 2010]超级钢琴(贪心+ST表+堆) 题面 给出一个长度为n的序列,选k段长度在L到R之间的区间,一个区间的值等于区间内所有元素之的和,使得k个区间的值之和最大.区 ...

  9. BZOJ 1507 Editor(块状链表)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1507 题意:一个文本编辑器,模拟以下操作: 思路:块状链表的主要操作: (1)find( ...

随机推荐

  1. Eclipse中的快捷键总结

    Eclipse中10个最有用的快捷键组合 一个Eclipse骨灰级开发者总结了他认为最有用但又不太为人所知的快捷键组合.通过这些组合可以更加容易的浏览源代码,使得整体的开发效率和质量得到提升.     ...

  2. css3爆炸效果更换图片轮播图

    思路:给一个div设置一个背景图片1.jpg,然后在这个div上面用两个for循环动态的创建一个列数为C行数为R数量的span,并给这些span设置宽高.定位并设置背景图片0.jpg,然后设置每个sp ...

  3. javascript使用栈结构将中缀表达式转换为后缀表达式并计算值

    1.概念 你可能听说过表达式,a+b,a+b*c这些,但是前缀表达式,前缀记法,中缀表达式,波兰式,后缀表达式,后缀记法,逆波兰式这些都是也是表达式. a+b,a+b*c这些看上去比较正常的是中缀表达 ...

  4. transition的局限

    transition的优点在于简单易用,但是它有几个很大的局限. (1)transition需要事件触发,所以没法在网页加载时自动发生. (2)transition是一次性的,不能重复发生,除非一再触 ...

  5. [转]iptables详解

    FROM : http://blog.chinaunix.net/uid-26495963-id-3279216.html 一:前言   防火墙,其实说白了讲,就是用于实现Linux下访问控制的功能的 ...

  6. Linux 进程与线程四(加锁--解锁)

    线程共享进程的内存空间,打开的文件描述符,全局变量. 当有多个线程同事访问一块内存空间或者一个变量.一个文件描述符,如果不加控制,那么可能会出现意想不到的结果. 原子操作 对于我们的高级语言(C语言, ...

  7. javarebel热部署 (转)

    Java web开发部署效率浅析 在进行java web程序开发过程中,经常遇到这种问题,修改一个java文件(*.java),需要重启web服务器(如tomcat,weblogic等),部署项目.而 ...

  8. 减少图片HTTP 请求的方案

    <Higb Performance Web Sites>(中文名:“高性能网站建设指南”)这本书对于前端工程师来说,绝对值得一读.本人有幸从公司借阅了,但不幸的是感觉翻译有点怪怪的.尤其是 ...

  9. 一款漂亮实用的Android开源日期控件timessquare

    这个开源控件可以兼容到SDK8版本,可以自定义显示的年月日,以及时间范围,如图 如果我们只想显示两个月的日期选择区间: final Calendar month = Calendar.getInsta ...

  10. 学习Shell脚本编程(第1期)_Shell命令行书写规则

    Shell命令行的书写规则 对Shell命令行基本功能的理解有助于编写更好的Shell程序,在执行Shell命令时多个命令可以在一个命令行上运行,但此时要使用分号(:)分隔命令,例如: [root@l ...