蒂花之秀 --- 大神用漫画讲解字符串匹配算法,让你噩梦变美梦
第二轮,我们把模式串后移一位,从主串的第二位开始,把主串和模式串的字符逐个比较:
data:image/s3,"s3://crabby-images/26da2/26da21367b68c4c8edda1db2df84d545c2ddcef6" alt=""
主串的第二位字符是 b,模式串的第二位字符也是 b,两者匹配,继续比较:
data:image/s3,"s3://crabby-images/8f657/8f657ab341f8e26204ebaeeb9793ffe29c246db1" alt=""
主串的第三位字符是 b,模式串的第三位字符也是 c,两者并不匹配。
第三轮,我们把模式串再次后移一位,从主串的第三位开始,把主串和模式串的字符逐个比较:
data:image/s3,"s3://crabby-images/dcd1b/dcd1b030a4c1f9ced8a42d76a6dc4944cd439273" alt=""
主串的第三位字符是 b,模式串的第三位字符也是 b,两者匹配,继续比较:
data:image/s3,"s3://crabby-images/00d32/00d326bed64063a7f3871a64b5463eb2a19f3f2a" alt=""
主串的第四位字符是 c,模式串的第四位字符也是 c,两者匹配,继续比较:
data:image/s3,"s3://crabby-images/8a999/8a999ee6b1ebed7c87fabb3081f146be8240c7dc" alt=""
主串的第五位字符是 e,模式串的第五位字符也是 e,两者匹配,比较完成!
由此得到结果,模式串 bce 是主串 abbcefgh 的子串,在主串第一次出现的位置下标是 2:
data:image/s3,"s3://crabby-images/c1612/c161222d021ffbe096419460b8c7b981e100404d" alt=""
以上就是小灰想出的解决方案,这个算法有一个名字,叫做 BF 算法,是 Brute Force(暴力算法)的缩写。
data:image/s3,"s3://crabby-images/28a2d/28a2d348f33c84bafbca592712bd8b6fbc9af7e6" alt=""
data:image/s3,"s3://crabby-images/f156c/f156c290ea31a62ae87b2f0bb428d152444943ea" alt=""
data:image/s3,"s3://crabby-images/07351/073519499d31728dee7e5ae36286a8211a8fa554" alt=""
上图的情况,在每一轮进行字符匹配时,模式串的前三个字符 a 都和主串中的字符相匹配,一直检查到模式串最后一个字符 b,才发现不匹配:
data:image/s3,"s3://crabby-images/9d885/9d8850b04d93c36c33ba9248eba249712ea2f88a" alt=""
这样一来,两个字符串在每一轮都需要白白比较 4 次,显然非常浪费。
假设主串的长度是 m,模式串的长度是 n,那么在这种极端情况下,BF 算法的最坏时间复杂度是 O(mn)。
data:image/s3,"s3://crabby-images/d24c8/d24c8e16a1fc2acb4f6b0fb24cb3bdd9ec58b560" alt=""
data:image/s3,"s3://crabby-images/910b1/910b1da90827035ed501583179fde4e49a19f696" alt=""
data:image/s3,"s3://crabby-images/777f0/777f03fefa641308e9b31b40a54cc883b5fd5961" alt=""
data:image/s3,"s3://crabby-images/8d873/8d873a948331822355f9bd1658f8a27d62554e51" alt=""
————————————
data:image/s3,"s3://crabby-images/bb0eb/bb0eb684a64e10ee9f2ee7c31b71a3f71e716dd0" alt=""
data:image/s3,"s3://crabby-images/5c69a/5c69a01101ab25ec4bb1e9181844d1df17500dca" alt=""
data:image/s3,"s3://crabby-images/b72e9/b72e9cb7da7d1a4ec8ecfbdf16024a187be2edbc" alt=""
data:image/s3,"s3://crabby-images/04fe1/04fe1997c4373505c91201255c0f6cbcacdfa348" alt=""
data:image/s3,"s3://crabby-images/e6a11/e6a1131fb8b5ce6678ca5030d5b0972df7a2e016" alt=""
data:image/s3,"s3://crabby-images/83eda/83eda85b33faeba41c224f83bd5d0be5baa380c4" alt=""
data:image/s3,"s3://crabby-images/3b57e/3b57ec25e2f9e5966f8c245a419c9ef9a6966f2d" alt=""
比较哈希值是什么意思呢?
用过哈希表的朋友们都知道,每一个字符串都可以通过某种哈希算法,转换成一个整型数,这个整型数就是 hashcode:
hashcode = hash(string)
显然,相对于逐个字符比较两个字符串,仅比较两个字符串的 hashcode 要容易得多。
data:image/s3,"s3://crabby-images/f6a55/f6a5599a0e8c83934b130e2b5d1de393a4d332f4" alt=""
data:image/s3,"s3://crabby-images/f7cab/f7cab809eb75264db724e5a6360048b0500858f7" alt=""
给定主串和模式串如下(假定字符串只包含 26 个小写字母):
data:image/s3,"s3://crabby-images/1f241/1f2410d41a80ef677dff01507c283ee1bf37d029" alt=""
第一步,我们需要生成模式串的 hashcode。
生成 hashcode 的算法多种多样,比如:
按位相加
这是最简单的方法,我们可以把 a 当做 1,b 当做 2,c 当做 3......然后把字符串的所有字符相加,相加结果就是它的 hashcode。
bce = 2 + 3 + 5 = 10
但是,这个算法虽然简单,却很可能产生 hash 冲突,比如 bce、bec、cbe 的 hashcode 是一样的。
转换成 26 进制数
既然字符串只包含 26 个小写字母,那么我们可以把每一个字符串当成一个 26 进制数来计算。
bce = 2*(26^2) + 3*26 + 5 = 1435
这样做的好处是大幅减少了 hash 冲突,缺点是计算量较大,而且有可能出现超出整型范围的情况,需要对计算结果进行取模。
为了方便演示,后续我们采用的是按位相加的 hash 算法,所以 bce 的 hashcode 是 10:
data:image/s3,"s3://crabby-images/2850d/2850ddd8cf91fe81714dadc5e022a93f7f79343c" alt=""
第二步,生成主串当中第一个等长子串的 hashcode。
由于主串通常要长于模式串,把整个主串转化成 hashcode 是没有意义的,只有比较主串当中和模式串等长的子串才有意义。
因此,我们首先生成主串中第一个和模式串等长的子串 hashcode,
即 abb = 1 + 2 + 2 = 5:
data:image/s3,"s3://crabby-images/f06b1/f06b1df2985340a58a60949125b15809bf15a8b9" alt=""
第三步,比较两个 hashcode。
显然,5!=10,说明模式串和第一个子串不匹配,我们继续下一轮比较。
第四步,生成主串当中第二个等长子串的 hashcode。
bbc = 2 + 2 + 3 = 7:
data:image/s3,"s3://crabby-images/ecba4/ecba4a07900b14cab6aacd04a0ab580c21102314" alt=""
第五步,比较两个 hashcode。
显然,7!=10,说明模式串和第二个子串不匹配,我们继续下一轮比较。
第六步,生成主串当中第三个等长子串的 hashcode。
bce= 2 + 3 + 5 = 10:
data:image/s3,"s3://crabby-images/b1fb4/b1fb4ab1dee5339fc0f63759d268afd37b16c59b" alt=""
第七步,比较两个 hashcode。
显然,10 ==10,两个 hash 值相等!这是否说明两个字符串也相等呢?
别高兴的太早,由于存在 hash 冲突的可能,我们还需要进一步验证。
第八步,逐个字符比较两字符串。
hashcode 的比较只是初步验证,之后我们还需要像 BF 算法那样,对两个字符串逐个字符比较,最终判断出两个字符串匹配。
data:image/s3,"s3://crabby-images/42428/424288f99f8cec3a21e9b699bb72ab4149010a0b" alt=""
最后得出结论,模式串 bce 是主串 abbcefgh 的子串,第一次出现的下标是 2。
data:image/s3,"s3://crabby-images/2a320/2a3205cc8984697293781f3dfa09c2d8de1737c6" alt=""
data:image/s3,"s3://crabby-images/3c208/3c208ff3b38d6c0c5ca42b1e78e9828298642d20" alt=""
什么意思呢?让我们再来看一个例子:
data:image/s3,"s3://crabby-images/93604/93604065b5351d21d12719729d953219a73884ee" alt=""
上图中,我已知子串 abbcefg 的 hashcode 是 26,那么如何计算下一个子串,也就是 bbcefgd 的 hashcode 呢?
data:image/s3,"s3://crabby-images/5ad2a/5ad2a836542326da16675b223760a65f79af8622" alt=""
我们没有必要把子串的字符重新进行累加运算,而是可以采用一个更简单的方法。由于新子串的前面少了一个 a,后面多了一个 d,所以:
新 hashcode = 旧 hashcode - 1 + 4 = 26-1+4 = 29
评论