「Sunday 算法」 是一种在字符串中查找子串的算法,是 Daniel M.Sunday 于1990年提出的字符串模式匹配算法。
Sunday 算法思想:对于给定文本串
T
与模式串p
,先对模式串p
进行预处理。然后在匹配的过程中,当发现文本串T
的某个字符与模式串p
不匹配的时候,根据启发策略,能够尽可能的跳过一些无法匹配的情况,将模式串多向后滑动几位。
Sunday 算法思想跟 Boyer Moore 算法思想类似。不同的是,Sunday 算法匹配顺序是从左向右,并且在模式串 p
匹配失败时关注的是文本串 T
中参加匹配的末尾字符的下一位字符。当文本串 T
中某个字符跟模式串 p
的某个字符不匹配时,可以将模式串 p
快速向右移动。
遇到不匹配字符时,可以根据以下两种情况向右快速进行移动:
- 情况 1:文本串
T
中与模式串p
尾部字符p[m - 1]
对应的字符下一个位置的字符T[i + m]
出现在模式串p
中。- 这种情况下,可将
T[i + m]
与模式串中最后一次出现的该字符对齐,如下图所示。 - 向右移动位数 = 文本串
T
中与模式串p
尾部位置的下一个位置 - T[i + m] 在模式串中最后一次出现的位置。 - 注意:文本串
T
中与模式串p
尾部位置的下一个位置其实就是「模式串长度」。
- 这种情况下,可将
- 情况 2:文本串
T
中与模式串p
尾部字符p[m - 1]
对应的字符下一个位置的字符T[i + m]
没有出现在模式串p
中。- 这种情况下,可将模式串整个右移,如下图所示。
- 向右移动位数 = 整个模式串长度 + 1。
整个 Horspool 算法步骤描述如下:
- 计算出文本串
T
的长度为n
,模式串p
的长度为m
。 - 先对模式串
p
进行预处理,生成后移位数表bc_table
。 - 将模式串
p
的头部与文本串T
对齐,将i
指向文本串开始位置,即i = 0
。j
指向模式串开始,即j = 0
,然后从模式串开始位置开始比较。- 如果文本串对应位置的字符
T[i + j]
与模式串对应字符p[j]
相同,则继续比较后一位字符。- 如果模式串全部匹配完毕,则返回模式串
p
在文本串中的开始位置i
。
- 如果模式串全部匹配完毕,则返回模式串
- 如果文本串对应位置的字符
T[i + j]
与模式串对应字符p[j]
不同,则:- 根据后移位数表
bc_table
和模式串末尾位置对应的文本串上的字符T[i + m]
,计算出可移动距离bc_table[T[i + m]]
,然后将模式串进行后移。
- 根据后移位数表
- 如果文本串对应位置的字符
- 如果移动到末尾也没有找到匹配情况,则返回
-1
。
生成后移位数表的代码实现比较简单,跟 Horspool 算法中生成后移位数表的代码差不多。具体步骤如下:
- 使用一个哈希表
bc_table
,bc_table[bad_char]
表示表示遇到坏字符可以向右移动的距离。 - 遍历模式串,以当前字符
p[i]
为键,可以向右移动的距离(m - i
)为值存入字典中。如果出现重复字符,则新的位置下标值会将之前存放的值覆盖掉。这样哈希表中存放的就是该字符在模式串中出现最右侧位置上的可向右移动的距离。
如果在 Sunday 算法的匹配过程中,如果 T[i + m]
不在 bc_table
中时,可令其为 m + 1
,表示可以将模式串整个右移到上一次匹配末尾后边两个位置上。如果 T[i + m]
在 bc_table
中时,可移动距离就是 bc_table[T[i + m]]
。这样就能计算出可以向右移动的位数了。
生成后移位数表的代码如下:
# 生成后移位数表
# bc_table[bad_char] 表示遇到坏字符可以向右移动的距离
def generateBadCharTable(p: str):
m = len(p)
bc_table = dict()
for i in range(m): # 迭代到最后一个位置 m - 1
bc_table[p[i]] = m - i # 更新遇到坏字符可向右移动的距离
return bc_table
# sunday 算法,T 为文本串,p 为模式串
def sunday(T: str, p: str) -> int:
n, m = len(T), len(p)
bc_table = generateBadCharTable(p) # 生成后移位数表
i = 0
while i <= n - m:
j = 0
if T[i: i + m] == p:
return i # 匹配完成,返回模式串 p 在文本串 T 的位置
if i + m >= n:
return -1
i += bc_table.get(T[i + m], m + 1) # 通过后移位数表,向右进行进行快速移动
return -1 # 匹配失败
# 生成后移位数表
# bc_table[bad_char] 表示遇到坏字符可以向右移动的距离
def generateBadCharTable(p: str):
m = len(p)
bc_table = dict()
for i in range(m): # 迭代到最后一个位置 m - 1
bc_table[p[i]] = m - i # 更新遇到坏字符可向右移动的距离
return bc_table
print(sunday("abbcfdddbddcaddebc", "aaaaa"))
print(sunday("abbcfdddbddcaddebc", "bcf"))
- Sunday 算法在平均情况下的时间复杂度为
$O(n)$ ,但是在最坏情况下时间复杂度会退化为$O(n * m)$ 。
- 【书籍】柔性字符串匹配 - 中科院计算所网络信息安全研究组 译
- 【博文】字符串模式匹配算法:BM、Horspool、Sunday、KMP、KR、AC算法 - schips - 博客园
- 【博文】字符串匹配——Sunday 算法 - Switch 的博客 - CSDN 博客