题面

传送门

题解

为什么大佬们全都是乱搞的……莫非这就是传说中的暴力能进队,乱搞能AC……

似乎有位大佬能有纯暴力+玄学优化\(AC\)(不算上\(uoj\)的\(Hack\)数据的话……这要是放到考场上就是切题的啊……)

整体思路呢,就是我们开一个线段树,线段树上的每一个区间维护“以这个区间右端点为结尾有可能成为后缀最小值的位置”

怎么合并呢

首先右儿子的所有节点都是可以加入的,因为它们后面也没有加上什么东西

然后对于左儿子来说它们相当于后面被整体怼了一个串,我们就要考虑它们是不是仍有可能成为后缀最小值了

首先比较两个位置的字典序大小,我们记\(u\)表示\(s[u,r]\),\(v\)表示\(s[v,r]\),假设\(u\)比\(v\)长,如果\(v\)不是\(u\)的前缀那么它们已经比较出了字典序大小,两个中肯定可以扔掉一个

然而如果\(v\)是\(u\)的前缀,事情就会变得比较辣手了

先给出结论:如果\(v\)是\(u\)的前缀,\(v\)就没有用了

为啥嘞?

因为它们都来自当前区间的左儿子,那么\(v\)的长度最短是\(r-mid+1\),\(u\)的长度最长是\(r-l+1\),根据我们正常线段树的写法\(v\)的长度绝对大于\(u\)的长度的一半

因为\(v\)的长度大于\(u\)的一半,同时\(v\)既是\(u\)的前缀也是它的后缀,那么我们发现\(u\)必定是一个有周期串,而且右儿子中至少有一个周期的开头\(w\)

所以现在\(w\)的字典序比它更小

那么加上新的字符之后\(v\)有没有可能变成后缀最小值呢?

答案是否定的,如果加的字符就是周期串上该有的字符,\(w\)比\(v\)小,如果加的字符小于该有的字符,\(w\)还是比\(v\)小,如果加的字符大于该有的字符,\(u\)就比\(v\)小,所以\(v\)无论如何都不可能再作为后缀最小值了

也就是说这个区间会把右儿子全都加上来,左端点最多加一个,那么一个节点上存的位置最多不会超过\(O(\log n)\)

所以我们只需要能在线段树上维护区间加以及判断字典序就行了

先说正解吧,判断字典序可以二分加哈希找到最后一个相等的位置再判断下一位的大小,不过我们要在区间加的状况下维护哈希值很困难,这样的话我们得把哈希值分块,一次修改就是\(O(\sqrt{n})\)了

正解的代码……去看\(shadowice\)巨巨的吧……才不是因为太长了不想打呢

于是这里区间加直接暴力,判断字典序也直接暴力……似乎是因为数据太水所以可以过……

而且我看了看\(uoj\)上这题排名前几的好像全都是打暴力的……跑得比正解快啊……

代码抄kcz的

//minamoto
#include<bits/stdc++.h>
#define R register
#define fp(i,a,b) for(R int i=(a),I=(b)+1;i<I;++i)
#define fd(i,a,b) for(R int i=(a),I=(b)-1;i>I;--i)
#define go(u) for(int i=head[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
using namespace std;
char buf[1<<21],*p1=buf,*p2=buf;
inline char getc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++;}
int read(){
R int res,f=1;R char ch;
while((ch=getc())>'9'||ch<'0')(ch=='-')&&(f=-1);
for(res=ch-'0';(ch=getc())>='0'&&ch<='9';res=res*10+ch-'0');
return res*f;
}
inline int getop(){R char ch;while((ch=getc())>'9'||ch<'0');return ch-'0';}
char sr[1<<21],z[20];int C=-1,Z=0;
inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
void print(R int x){
if(C>1<<20)Ot();if(x<0)sr[++C]='-',x=-x;
while(z[++Z]=x%10+48,x/=10);
while(sr[++C]=z[Z],--Z);sr[++C]='\n';
}
const int N=2e5+5;
int s[N],n,q;
inline int cmp(const int &a,const int &b,const int &r){
fp(i,0,r-max(a,b))if(s[a+i]!=s[b+i])return s[a+i]<s[b+i]?-1:1;
return 0;
}
struct node{
node *lc,*rc;int pos[21],cnt,l,r;
inline void ins(R int x){pos[++cnt]=x;}
void upd(){
fp(i,1,rc->cnt)pos[i]=rc->pos[i];
cnt=rc->cnt;
fp(i,1,lc->cnt)
while(233){
int t=cmp(lc->pos[i],pos[cnt],r);
if(t>0)break;
if(!t){
(((r-pos[cnt]+1)<<1)>r-lc->pos[i]+1)?--cnt:0,
pos[++cnt]=lc->pos[i];
break;
}
if(!--cnt){
pos[++cnt]=lc->pos[i];
break;
}
}
}
}pool[N<<2],*rt;
int tot,ql,qr,res,d;
inline node* newnode(){return &pool[tot++];}
void build(node* &p,int l,int r){
p=newnode(),p->l=l,p->r=r;
if(l==r)return p->ins(l),void();
int mid=(l+r)>>1;
build(p->lc,l,mid),build(p->rc,mid+1,r);
p->upd();
}
void update(node* p){
if(ql<=p->l&&qr>=p->r)return;
if(ql<=p->lc->r)update(p->lc);
if(qr>p->lc->r)update(p->rc);
p->upd();
}
void query(node* p){
if(ql<=p->l&&qr>=p->r){
R int i=1;!res?res=p->pos[i++]:0;
for(;i<=p->cnt;++i)cmp(p->pos[i],res,qr)<0?res=p->pos[i]:0;
return;
}
if(qr>p->lc->r)query(p->rc);
if(ql<=p->lc->r)query(p->lc);
}
int main(){
// freopen("testdata.in","r",stdin);
// freopen("qwq.out","w",stdout);
n=read(),q=read();
fp(i,1,n)s[i]=read();
build(rt,1,n);
while(q--)if(getop()&1){
ql=read(),qr=read(),d=read();
if(qr-ql+1<=(n>>1))fp(i,ql,qr)s[i]+=d;
else{
fp(i,1,ql-1)s[i]-=d;
fp(i,qr+1,n)s[i]-=d;
}
update(rt);
}else{
ql=read(),qr=read(),res=0;
query(rt),print(res);
}
return Ot(),0;
}

洛谷P5211 [ZJOI2017]字符串(线段树+乱搞)的更多相关文章

  1. 【BZOJ】1012: [JSOI2008]最大数maxnumber /【洛谷】1198(线段树)

    Description 现在请求你维护一个数列,要求提供以下两种操作:1. 查询操作.语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值.限制:L不超过当前数列的长度.2. 插 ...

  2. 洛谷题解P4314CPU监控--线段树

    题目链接 https://www.luogu.org/problemnew/show/P4314 https://www.lydsy.com/JudgeOnline/problem.php?id=30 ...

  3. 洛谷P3372/poj3468(线段树lazy_tag)(询问区间和,支持区间修改)

    洛谷P3372 //线段树 询问区间和,支持区间修改 #include <cstdio> using namespace std; struct treetype { int l,r; l ...

  4. 洛谷P4065 [JXOI2017]颜色(线段树)

    题意 题目链接 Sol 线段树板子题都做不出来,真是越来越菜了.. 根据题目描述,一个合法区间等价于在区间内的颜色没有在区间外出现过. 所以我们可以对于每个右端点,统计最长的左端点在哪里,刚开始以为这 ...

  5. 洛谷P5111 zhtobu3232的线段树

    题意:给定线段树,上面若干个节点坏了,求能表示出多少区间. 区间能被表示出当且仅当拆出来的log个节点都是好的. 解:每个区间在最浅的节点处计算答案. 对于每个节点维护从左边过来能有多少区间,从右边过 ...

  6. 洛谷P3960 列队 NOIp2017 线段树/树状数组/splay

    正解:动态开点线段树 解题报告: 传送门! 因为最近学主席树的时候顺便get到了动态开点线段树?刚好想起来很久很久以前就想做结果一直麻油做的这题,,,所以就做下好了QAQ 然后说下,这题有很多种方法, ...

  7. 题解——洛谷P2781 传教(线段树)

    可以说是数据结构学傻了的典型案例了 昨天跳到这题上 然后思考了一下 噫!好!线段树裸题 然后打完板子,发现\(  n \le 10^9 \) 显然线段树直接做不太行 然后这题又只有普及的难度 然后我就 ...

  8. 洛谷P4198 楼房重建(线段树)

    题意 题目链接 Sol 别问我为什么发两遍 就是为了骗访问量 这个题的线段树做法,,妙的很 首先一个显然的结论:位置\(i\)能被看到当且仅当\(\frac{H_k}{k} < \frac{H_ ...

  9. 2018.08.16 洛谷P1471 方差(线段树)

    传送门 线段树基本操作. 把那个方差的式子拆开可以发现只用维护一个区间平方和和区间和就可以完成所有操作. 同样区间修改也可以简单的操作. 代码: #include<bits/stdc++.h&g ...

随机推荐

  1. mysql存储过程(procedure)

    #创建带参数的存储过程 delimiter // ),out p int) begin ; end // delimiter call pro_stu_name_pass(@n,@p); select ...

  2. 用vim生成一批递增ID

    假设说要生成1000个以xxx开头的后面加数字的ID,比如xxx1到xxx1000.一般我们可以通过.csv去递增,然后替换,但是直接用vim也是可以达到这样的目的. 下面通过一个gif图演示这个过程 ...

  3. java算法 蓝桥杯算法训练 Fibonacci数列

    问题描述 Fibonacci数列的递推公式为:Fn=Fn-1+Fn-2,其中F1=F2=1. 当n比较大时,Fn也非常大,现在我们想知道,Fn除以10007的余数是多少. 输入格式 输入包含一个整数n ...

  4. go_内建变量类型

    bool, string (u)int, (u)int8, (u)int16, (u)int32, (u)int64, uintptr (uintptr 是指针) byte, rune(表示字符cha ...

  5. 【LA3211 训练指南】飞机调度 【2-sat】

    题意 有n嫁飞机需要着陆.每架飞机都可以选择“早着陆”和“晚着陆”两种方式之一,且必须选择一种.第i架飞机的早着陆时间为Ei,晚着陆时间为Li,不得在其他时间着陆.你的任务是为这些飞机安排着陆方式,使 ...

  6. PHP异常处理详解

      PHP异常处理详解 异常处理(又称为错误处理)功能提供了处理程序运行时出现的错误或异常情况的方法. 异常处理通常是防止未知错误产生所采取的处理措施.异常处理的好处是你不用再绞尽脑汁去考虑各种错误, ...

  7. [erlang 002]gen_server中何时会跑到terminate函数

    一.从start方法产出的独立gen_server进程 实验代码: %%%-------------------------------------- %%% @Module  : %%% @Auth ...

  8. Cloudstack4.2之改变数据卷容量的大小(Resize Data Volumes)

    下图标注了这个功能在cloudstack4.2 UI中的位置 在cloudstack中是通过磁盘服务来设定卷的大小的.管理员可以设置相应的磁盘服务以供用户来使用.为了增强系统的灵活性,方便最终用户的使 ...

  9. How to Get the Length of File in C

    How to get length of file in C //=== int fileLen(FILE *fp) { int nRet = -1; int nPosBak; nPosBak = f ...

  10. 认证服务号Thinkphp微信支付

    公众号配置 1.微信支付过程中需要获取用户的openid,所以,仍需填写 网页授权域名 2.微信支付授权目录  Thinkphp目录格式为www.xxx.cn/home/wxpay/ 这里目录不能填写 ...