其实SA这个东西很久之前就听过qwq

但是基本已经忘的差不多了

嘤嘤嘤

QWQ感觉自己不是很理解啊

所以写不出来那种博客

QWQ只能安利一些别人的博客了

真的是讲的非常好

不要在意名字

orz,膜拜他们

顺便弄上自己的代码(里面有一些需要注意的地方)

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk makr_pair
#define ll long long
using namespace std;
inline int read()
{
int x=0,f=1;char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
const int maxn = 2e6+1e2;
int wb[maxn];
int rk[maxn];
int sa[maxn],tmp[maxn];
char a[maxn];
int h[maxn],height[maxn];
int n;
void getsa()
{
int *x=rk,*y=tmp;
int s = 128;
for (int i=1;i<=n;i++) x[i]=a[i],y[i]=i; //初始每个长度为1的后缀的rank是他自己的字符大小,第二关键字相当于空,那么就顺次赋值为i
for (int i=1;i<=s;i++) wb[i] =0;
for (int i=1;i<=n;i++) wb[x[y[i]]]++; // 这里其实基数排序的时候,x表示上一轮的rank,y[i]表示第二关键字排名为i的第一关键字的位置是多少
for (int i=1;i<=s;i++) wb[i]+=wb[i-1];//做前缀和就能更好的算出来排名,比如说有3个a,2个b,那么自然第一个b的排名就要从4开始
for (int i=n;i>=1;i--) sa[wb[x[y[i]]]--]=y[i]; //只能感性理解了啊qwq之所以倒着枚举是为了保证在第一关键字相同的时候,第二关键字也是有序的
int p = 0;
for (int j=1;p<n;j<<=1) //p是指本质不同的串的个数
{
//x表示上一轮的rank
//y表示排名为i的第二关键字的第一关键字的位置是多少(空优先)
p=0;
//这里可以这么理解,如果一个串他的位置是大于n-j+1,那么他一定是没有第二关键字的。
for (int i=n-j+1;i<=n;i++) y[++p]=i; //第二关键字为空,就排名靠前
for (int i=1;i<=n;i++) if (sa[i]>j) y[++p]=sa[i]-j; //如果排名为i的位置是大于j的,那么他可以成为一个第二关键字,并且第一关键字的位置应该是sa[i]-j;
for (int i=1;i<=s;i++) wb[i]=0;
for (int i=1;i<=n;i++) wb[x[y[i]]]++;
for (int i=1;i<=s;i++) wb[i]+=wb[i-1];
for (int i=n;i>=1;i--) sa[wb[x[y[i]]]--]=y[i]; //这里i之所以从n开始,因为我们要保证排序第一关键字的时候,第二关键字一定也是符合原来的顺序的,就是说,原来第二关键字大的,一定在后面(这个是基数排序的思想)
swap(x,y);//交换之后,y表示上一轮的rank,x是一个新的数组
p=1;
x[sa[1]]=1;
//若两个串的两部分在上一轮rank都相等的话, 那么无法分辨,所以p不用加
for (int i=2;i<=n;i++) x[sa[i]] = (y[sa[i]]==y[sa[i-1]] && y[sa[i-1]+j]==y[sa[i]+j]) ? p : ++p;
s=p;
}
for (int i=1;i<=n;i++) rk[sa[i]]=i;
h[0]=0;
for (int i=1;i<=n;i++)
{
//h[i]表示i号后缀与它前一名的后缀的最长公共前缀
//height[i]表示排名为i的后缀和排名为i-1的后缀的lcp
h[i]=max(h[i-1]-1,0);
while (i+h[i]<=n && sa[rk[i]-1]+h[i]<=n && a[i+h[i]]==a[sa[rk[i]-1]+h[i]])
h[i]++;
}
for (int i=1;i<=n;i++) height[i]=h[sa[i]];
}
int main()
{
scanf("%s",a+1);
n=strlen(a+1);
getsa();
for (int i=1;i<=n;i++) cout<<sa[i]<<" ";
return 0;
}

Update

整理一些\(SA\)的小性质和经典应用。(会持续更新的)

1.求两个后缀的\(lcp\) ,应该是\(min(height[rk[i]+1],height[rk[i]+2].....height[rk[j]])\)

2.对于排名为\(i\)的后缀,与它\(lcp\)最长的后缀应该是排名为\(i-1\),(可以理解为越靠前差异越多,越靠前,取\(min\)的区间就越长)

3.最长可重叠重复子串,应该是\(max(height[i])\)(这里把子串看成后缀的前缀,同时依据性质2就能得到)

4.给定一个子串,求不相同子串的个数,这里要这么考虑,按照字典序加入,每加入一个字符串,会新增加\(n-sa[i]+1\)个新的子串,但是会重复\(height[i]\)个,\((只有lcp会重复,同时依据性质2)\)

5.给定两个串,求他们的最长公共子串。

将B串拼到A串后面,然后中间添加一个非法字符,然后直接想询问最大的lcp(保证\(sa[i]和sa[i-1]\)分别位于两个串即可)

6.给定两个串,求他们的公共子串数目。

将B串拼到A串后面,然后中间添加一个非法字符,然后对于每个\(height\)用单调栈维护出左右最远能扩展到哪里。然后\(ans\)加上\(height[i]*(geta(i-1,l[i]-1)*getb(r[i],i)+getb(i-1,l[i]-1)*geta(r[i],i))\)

这里之所以是这个式子的原因(第一要保证是一个端点属于A串,一个属于B串。另一个原因是因为对于一个扩展区间\([l,pos,r]\)来说,选择后缀的右端点是在\([pos,r]\)而左端点是\([l-1,pos-1]\),因为后缀的选择的左边对于\(height\)是开区间,参考性质1。

洛谷3809 SA模板 后缀数组学习笔记(复习)的更多相关文章

  1. 洛谷-P3809-后缀排序(后缀数组)

    看了求后缀数组的倍增法之后很快就理解了,但是自己写的倍增法用map排序还是超时了.然后看了两天别人写的模板,题目是通过了,但感觉代码还是半懂半背的.以后多熟悉熟悉吧: 后缀数组 #include &q ...

  2. 洛谷P3763 [TJOI2017]DNA(后缀数组 RMQ)

    题意 题目链接 Sol 这题打死我也不会想到后缀数组的,应该会全程想AC自动机之类的吧 但知道这题能用后缀数组做之后应该就不是那么难了 首先把\(S\)和\(S0\)拼到一起跑,求出Height数组 ...

  3. 洛谷 P4143 采集矿石 后缀数组

    题目背景 ZRQ 成功从坍塌的洞穴中逃了出来.终于,他看到了要研究的矿石.他想挑一些带回去完成任务. 题目来源:Zhang_RQ哦对了 \(ZRQ\) 就他,嗯 题目描述 ZRQ 发现这里有 \(N\ ...

  4. 洛谷P5108 仰望半月的夜空(后缀数组)

    题意 题目链接 Sol warning:下面这个做法只有95分,本地拍了1w+组都没找到错误我表示十分无能为力 我们考虑每个串的排名去更新答案,显然排名为\(1\)的后缀的前缀一定是当前长度的字典序最 ...

  5. 【洛谷P3369】普通平衡树——Splay学习笔记(一)

    二叉搜索树(二叉排序树) 概念:一棵树,若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值: 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值: 它的左.右子树也分别为二叉搜索树 ...

  6. 【洛谷P3391】文艺平衡树——Splay学习笔记(二)

    题目链接 Splay基础操作 \(Splay\)上的区间翻转 首先,这里的\(Splay\)维护的是一个序列的顺序,每个结点即为序列中的一个数,序列的顺序即为\(Splay\)的中序遍历 那么如何实现 ...

  7. [洛谷P3809]【模板】后缀排序

    [洛谷P3809][模板]后缀排序 题目大意: 对于给定的长度为\(n(n\le10^6)\)的字符串求后缀数组\(sa[i]\). 思路: 倍增+快排构造后缀数组.代码参考<挑战程序设计竞赛& ...

  8. 洛谷 P3919 【模板】可持久化数组(可持久化线段树/平衡树)-可持久化线段树(单点更新,单点查询)

    P3919 [模板]可持久化数组(可持久化线段树/平衡树) 题目背景 UPDATE : 最后一个点时间空间已经放大 标题即题意 有了可持久化数组,便可以实现很多衍生的可持久化功能(例如:可持久化并查集 ...

  9. 洛谷 P3387 【模板】缩点 DAGdp学习记

    我们以洛谷P3387 [模板]缩点 来学习DAGdp 1.这道题的流程 //伪代码 for i->n if(i未被遍历) tarjan(i) 缩点() DAGdp() 完成 首先tarjan这部 ...

随机推荐

  1. Gogs (Go git server) 使用笔记

    issue: 话题,一个新特性,BUG或其他关注的任何话题,都可创建issure,便于讨论,明确目标. label: 标签,一般用于描述issue的类型,如:bug.feature.enhanceme ...

  2. 笔记:如何使用postgresql做顺序扣减库存

    如何使用postgresql做顺序扣减库存 Ⅰ.废话在前面 首先这篇笔记源自于最近的一次需求,这个临时性需求是根据两份数据(库存数据以及出库数据) 算出实际库存给到业务,至于库存为什么不等于剩余库存, ...

  3. 整理之Activity

    基础 生命周期 执行层次 进 退 创建与销毁 onCreate() onDestroy() 是否可见 onStart() onStop() 是否在前台(可交互) onResume() onPause( ...

  4. 2021 年 9 月 TIOBE 指数 C# 增长突破 1.2%

    TIOBE 编程社区指数是编程语言流行程度的指标.该指数每月更新一次.评级基于全球熟练工程师.课程和第三方供应商的数量.谷歌.必应.雅虎.维基百科.亚马逊.YouTube 和百度等流行搜索引擎用于计算 ...

  5. SNMP协议之序言

    最近两周公司分配一个任务:使用snmp协议做一个网管,来配置我们的产品.这可以说是我第一次听说这个协议,我问了一下周围的同事这是个什么协议,同事说"简单网络管理协议",其实这个协议 ...

  6. GDB调试:Linux开发人员必备技能

    开篇词:Linux C/C++ 开发人员要熟练掌握 GDB 调试 大家好,我是范蠡,目前在某知名互联网旅游公司基础框架业务部技术专家组任开发经理一职. 本系列课程的主题是 Linux 后台开发的 C/ ...

  7. js 显示日期时间,时间过一秒加1

    html: <div id="data"><font>2017年10月17日 15:11:11</font></span> js: ...

  8. go语言游戏服务端开发(二)——网络通信

    一.网络层 网络游戏客户端除了全局登录使用http请求外,一般通过socket长连接与服务端保持连接.go语言的net包提供网络socket长连接相关操作. 对于服务端,一般经历 Listen.Acc ...

  9. node实战小例子

    第一章 2020-2-6 留言小本子 思路(由于本章没有数据库,客户提交的数据放在全局变量,接收请求用的是bodyParser, padyParser使用方法 app.use(bodyParser.u ...

  10. Linux上安装服务器监视工具,名为pyDash。

    pyDash – A Web Based Linux Performance Monitoring Tool 你可以通过以下命令来判断是否已安装: pip --version # Python2.x ...