这篇博客不打算讲多么详细,网上关于后缀数组的blog比我讲的好多了,这一篇博客我是为自己加深印象写的。

给你们分享了那么多,容我自私一回吧~

参考资料:这位dalao的blog

一、关于求SuffixArray的一些变量定义:

1. sa[i]=j,表示第i名的后缀从j开始

**存的是下标**

2. rnk[i]=j,从i开始的后缀是第j名的

**与sa为互逆运算,存的是值**

3. tp[i]=j, 第二关键字为i的后缀从j开始

**可理解为第二关键字的SA,存的是下标**

插入解释一下第一关键字和第二关键字:

我们要对所有的后缀进行排序,怎么排呢?

开始时,我们每个字符的后缀存的只有它自己,所以它后缀的大小就是它的ASCII码。

我们把每个字符i看成(s[i],i)的二元组,如果我们直接丢pair<int,int>里面然后std::sort,

这样的时间复杂度是O(log^2 n)的,显然不够优秀。

所以就需要用到基数排序RadixSort,不了解的自行百度。

再使用倍增法,就可以使我们排序的时间复杂度降低到O(logn)。

所以我们要对每个后缀的前两个字母进行排序,第一个字母的相对关系已经得到了。

第i个后缀的第二个字母,就是第i+1个后缀的第一个字母,利用这个关系我们第二个字母的相对关系也就知道了。

我们的tp数组就是用来记录它的,rnk[i]表示上一轮中第i个后缀的排名。

这里引用神仙attack的一句话,我觉得讲的非常到位:

对于一个长度为w的后缀,你可以形象的理解为:

第一关键字针对前w2个字符形成的字符串,第二关键字针对后w2个字符形成的字符串

然后对每个后缀的前4个字母组成的字符串排序,前8个,前16个...这就是倍增法求SA的流程了。

给出RadixSort的代码:

  1. void RadixSort(int a[],int b[]){//基数排序
  2. for(int i=;i<=m;i++)tax[i]=;
  3. for(int i=;i<=n;i++)tax[a[i]]++;
  4. for(int i=;i<=m;i++)tax[i]+=tax[i-];
  5. for(int i=n;i>=;i--)sa[tax[a[b[i]]]--]=b[i];
  6. }

实在不能理解RadixSort也没有关系,代码很短

再给出求SA的代码:

  1. bool cmp(int *r,int a,int b,int k){
  2. return r[a]==r[b]&&r[a+k]==r[b+k];
  3. }
  4. void getSA(int a[],int b[]){
  5. for(int i=;i<=n;i++)
  6. m=max(m,a[i]=s[i]-''),b[i]=i;
  7. RadixSort(a,b);
  8. for(int p=,j=;p<n;j<<=,m=p){
  9. p=;
  10. for(int i=;i<=j;i++)b[++p]=n-j+i;
  11. for(int i=;i<=n;i++)if(sa[i]>j)b[++p]=sa[i]-j;
  12. RadixSort(a,b);
  13. int *t=a;a=b;b=t;
  14. a[sa[]]=p=;
  15. for(int i=;i<=n;i++)
  16. a[sa[i]]=cmp(b,sa[i],sa[i-],j)?p:++p;
  17. }
  18. }

关于代码的解释,有时间再填坑。本蒟蒻要学的算法还很多...SA就粗略地理解一下好了

开始填坑,先补充一个东西叫height数组。

height[i]表示排名为i的后缀和排名为i-1的后缀的最长公共前缀LCP。

暴力求解时间复杂度是O(n^2),根据一个性质height[i+1]>=height[i]-1

可以O(n)时间内求出height数组,具体代码:

  1. void getHeight(){
  2. for(int i=,j=;i<=n;i++){
  3. if(j)j--;
  4. while(s[i+j]==s[sa[rnk[i]-]+j])j++;
  5. height[rnk[i]]=j;
  6. }
  7. }

关于这个height数组,它可以干什么,给出一张列表:

两个后缀的最大公共前缀

lcp(x,y)=min(heigh[x−y])lcp(x,y)=min(heigh[x−y]), 用rmq维护,O(1)查询

可重叠最长重复子串

height数组里的最大值

不可重叠最长重复子串

首先二分答案x,对height数组进行分组,保证每一组的最小height都>=x

依次枚举每一组,记录下最大和最小长度,若sa[max]−sa[min]>=x那么可以更新答案

本质不同的子串的数量

枚举每一个后缀,第i个后缀对答案的贡献为len−sa[i]+1−height[i]

浅谈后缀数组SA的更多相关文章

  1. 后缀数组(SA)总结

    后缀数组(SA)总结 这个东西鸽了好久了,今天补一下 概念 后缀数组\(SA\)是什么东西? 它是记录一个字符串每个后缀的字典序的数组 \(sa[i]\):表示排名为\(i\)的后缀是哪一个. \(r ...

  2. 后缀数组SA学习笔记

    什么是后缀数组 后缀数组\(sa[i]\)表示字符串中字典序排名为\(i\)的后缀位置 \(rk[i]\)表示字符串中第\(i\)个后缀的字典序排名 举个例子: ababa a b a b a rk: ...

  3. 后缀数组SA入门(史上最晦涩难懂的讲解)

    参考资料:victorique的博客(有一点锅无伤大雅,记得看评论区),$wzz$ 课件(快去$ftp$%%%),$oi-wiki$以及某个人的帮助(万分感谢!) 首先还是要说一句:我不知道为什么我这 ...

  4. bzoj3796(后缀数组)(SA四连)

    bzoj3796Mushroom追妹纸 题目描述 Mushroom最近看上了一个漂亮妹纸.他选择一种非常经典的手段来表达自己的心意——写情书.考虑到自己的表达能力,Mushroom决定不手写情书.他从 ...

  5. [笔记]后缀数组SA

    参考资料这次是真抄的: 1.后缀数组详解 2.后缀数组-学习笔记 3.后缀数组--处理字符串的有力工具 定义 \(SA\)排名为\(i\)的后缀的位置 \(rk\)位置为\(i\)的后缀的排名 \(t ...

  6. 【字符串】后缀数组SA

    后缀数组 概念 实际上就是将一个字符串的所有后缀按照字典序排序 得到了两个数组 \(sa[i]\) 和 \(rk[i]\),其中 \(sa[i]\) 表示排名为 i 的后缀,\(rk[i]\) 表示后 ...

  7. 后缀数组SA

    复杂度:O(nlogn) 注:从0到n-1 const int maxn=1e5; char s[maxn]; int sa[maxn],Rank[maxn],height[maxn],rmq[max ...

  8. 洛谷2408不同字串个数/SPOJ 694/705 (后缀数组SA)

    真是一个三倍经验好题啊. 我们来观察这个题目,首先如果直接整体计算,怕是不太好计算. 首先,我们可以将每个子串都看成一个后缀的的前缀.那我们就可以考虑一个一个后缀来计算了. 为了方便起见,我们选择按照 ...

  9. 洛谷4248 AHOI2013差异 (后缀数组SA+单调栈)

    补博客! 首先我们观察题目中给的那个求\(ans\)的方法,其实前两项没什么用处,直接\(for\)一遍就求得了 for (int i=1;i<=n;i++) ans=ans+i*(n-1); ...

随机推荐

  1. Shiro RememberMe 1.2.4远程代码执行漏洞-详细分析

    本文首发于先知: https://xz.aliyun.com/t/6493 0x01.漏洞复现 环境配置 https://github.com/Medicean/VulApps/tree/master ...

  2. 攻防世界Hello,CTF writeup

    解题过程 首先在ida中进行反汇编,查看main函数的代码: 代码的的36行处进行了一个字符串比较,如果v10的值等于v13的值会反馈一个success的输出.v13的值在第15行给出,因此需要知道v ...

  3. 讨论SQLite数据库损坏与修复

      版权声明:博客将逐步迁移到 http://cwqqq.com https://blog.csdn.net/cwqcwk1/article/details/45541409 昨晚,朋友和我反馈SQL ...

  4. python md5验签

    import hashlib #api验签 参数按首字母排序,然后拼接clientid=123456&num=xxxx&status=1&timestamp=157319776 ...

  5. 使用mybatis的resultMap进行复杂查询

        记录下mybatis的集合查询中碰到的问题 https://jaychang.iteye.com/blog/2357143   MyBatis ofType和javaType区别 https: ...

  6. Docker使用Portainer搭建可视化界面

    Portainer介绍 Portainer是Docker的图形化管理工具,提供状态显示面板.应用模板快速部署.容器镜像网络数据卷的基本操作(包括上传下载镜像,创建容器等操作).事件日志显示.容器控制台 ...

  7. 当微信小程序遇到AR(二)

    当微信小程序遇到AR,会擦出怎么样的火花?期待与激动...... 通过该教程,可以从基础开始打造一个微信小程序的AR框架,所有代码开源,提供大家学习. 本课程需要一定的基础:微信开发者工具,JavaS ...

  8. “但行好事 莫问前程 只问耕耘 不问收获 成功不必在我 而功力必不唐捐” 科技袁人·年终盛典——5G是科技时代非常重要的基础设施

    中国的科技实力:用数据对比展示当前中国整体科技实力在国际中的发展水平和未来的发展趋势. 主要分为基础研究和应用研究.其中基础研究通过论文数据进行对比展示,应用研究通过发明专利数据. 又分别结合当今中国 ...

  9. LeetCode刷题1——只出现一次的数字

    一.题目要求 二.题目背景 位运算:或,异或,与,移位 三.解题思路 (1)要求算法时间复杂度是线性的,O(n),想到的是先将列表排序,排序后相同的数值两两之间前后相邻,进行偶数次循环,判断两两数值是 ...

  10. 【VS开发】千兆以太网的传输速度

    千兆以太网主流标准 千兆以太网络技术早在上世纪90年代末就已成熟,其中,1995年国际标准化组织TIA/EIA颁布了1000Base-TX标准,该标准的目的是把双绞线用于千兆以太网中,其目的是在6类非 ...