KMP算法(2)

return -1;

}

if (t.length() < p.length()) {

return -1;

}

char[] ct = t.toCharArray();

char[] cp = p.toCharArray();

for (int i = 0; i < ct.length - cp.length + 1; i++) {

for (int j = 0; j < cp.length; j++) {

if (ct[i + j] == cp[j]) {

if (j == cp.length - 1) {

// 已经完全匹配

return i;

}

} else {

break;

}

}

}

return -1;

}

素朴算法的猜想


主串的每个字符都要与匹配串进行比较,其实并没有必要

a与b不匹配,匹配串每次移动一个位置,简称滑动,已经明知道匹配串的第一个位置的字符b不匹配主串的第1与第2位置的字符a,如果直接匹配b,跳过主串不匹配的字符。

前缀与后缀


那么匹配串需要滑动多少才能和当前主串的匹配位置相等,在明白前,先搞清楚前缀与后缀的概念

前缀:除了最后一个字母外,一个字符串的全部实现组合。

后缀:除了第一个字母外,一个字符串的全部实现组合

比如:abababa

前缀:a、ab、aba、abab、ababa、ababab

后缀:a、ba、aba、baba、ababa、bababa

加入有这样的一个主串T : babaabababada,匹配串是P : abababa

​ 首先T[0]与P[0]不匹配,并且下一个T[1]与P[1]是匹配的,按照人类的想法,就是把P向右移动一位。

T[1]== P[0]、T[2] == P[1]、T[3] == P[2] ,但是T[4] ≠ P[3]

按照之前的朴素算法,P这个时候就要向右移动一位,并且i指针又开始回去了,做了重复的计算。

i指针可以不用向前回退,那就要让j指针回退。上图中T[4] ≠ P[3],但是P[3]之前的肯定是匹配的,就是aba,其前缀是a、ab;后缀是ba、a,前缀与后缀可以重合的就只有a,那么最大可重合元素数就是1,

为什么要计算这个最大重合数?

假如有这样的一个字符串,abab,其前缀:a、ab、aba;后缀bab、ab、b;可以重合的就是ab,那么最大可重合数就是2.也就是说abab向右移动2位,前缀与后缀就可以重合。理解这个非常重要,理解这个就理解了kmp算法的一半原理。

接下来就只要知道一个字符串的所有前缀表就可以知道需要移动的位数了,手动计算abababa的前缀表

假如abababa中只有最后的一个字符a没有匹配上,那么也就是说ababab是匹配的。

ababab 前缀:a、ab、aba、abab、ababa;后缀:babab、abab、baba、ab、b;最大重合数:4

这样操作的含义:

当前字符不匹配,就去找当前字符的前面字串的最大重合数,例如上面例子中的4 ,P[6]与T[i]不匹配,j指针回溯到4,即拿T[4]与P[i]比较,前面的abab就可以与主串达到最大匹配数量4

没有看懂?我再举个例子

假设有一个匹配串abad

在T[3]这里不匹配,找到前面的字串aba,aba的前缀:a、ab;后缀ba、a;最大重合数是1(就是首尾的a),j回溯到1后与i比较,此时j前面的子串已经达到和T重合的最大数量。注意i可没有回溯。

也就是说,如果j位置不匹配,就寻找j前面的子串的最大重合数,然后将j回溯到这个最大值处。此时就需要重建一个前缀表

前缀表next


接下来加计算前缀表

假设有匹配串 abaabbabaa

如果 j = 0 处不匹配,后面也没必要比较了,i向前增加,注意这是一个重要的跳转条件,next[0]=-1,-1是一个默认给定的值;

如果 j = 1 处不匹配,子串a的前缀与后缀重合的最大值为0, next[1]=0;

如果j = 2 处不匹配,子串ab的前缀与后缀重合的最大值为0, next[1]=0;

如果j =3 处不匹配,子串aba的前缀与后缀重合的最大值为1 next[2]=1;

如果j =4 处不匹配,子串abaa的前缀与后缀重合的最大值为1 next[3]=1;

如果j =5 处不匹配,子串abaab的前缀与后缀重合的最大值为2 next[4]=2;

如何写好这个逻辑的代码?需要找规律,其实就2钟:

  1. P[j + 1 ] == P[next[j]]

比如这种情况 abab,已知 j = 3, next[j - 1] = next[2] = 1,P[next[2]] =P[1] = b; 求next[j]

解: next[j] = next[j-1] + 1

因为aba的最大重合数是1,也就是在b的位置,现在在后面添加一个b,这个b和P[1]的值b是一样的,必然会将重合数增1。

  1. P[j + 1] ≠ next[j]
  • P[0] == P[j + 1]

首尾字符相等,那么重合数只能是1,即next[j] = 0

  • P[0] ≠ P[j + 1]

首尾字符不相等,前缀与后缀不会再有重合这种情况,即next[j] = 0

经过上面的解释,我们用前面的next值推导出后面的。

如果 j =6 处不匹配,P[5] = b;next[4] =2, P[2] = a, 因为P[5] ≠ P[2],且P[0] ≠ P[5] , 则 next[5]=0;

如果 j =7 处不匹配,P[6] = a;next[5] =0, P[0] = a, 因为P[6] == P[0] , 则 next[6]=next[5] +1 = 1;

如果 j =8 处不匹配,P[7] = b;next[6] =1, P[1] = b, 因为P[7] == P[1] , 则 next[7]=next[6] +1 = 2;

如果 j =9 处不匹配,P[8] = a;next[7] =2, P[2] = a, 因为P[8] == P[2] , 则 next[8]=next[7] +1 = 3;

public static int[] getNext(String ps) {

char[] p = ps.toCharArray();

int[] next = new int[p.length];

next[0] = -1;// 默认给给定的值

面试复习路线,梳理知识,提升储备

自己的知识准备得怎么样,这直接决定了你能否顺利通过一面和二面,所以在面试前来一个知识梳理,看需不需要提升自己的知识储备是很有必要的。

关于知识梳理,这里再分享一下我面试这段时间的复习路线:(以下体系的复习资料是我从各路大佬收集整理好的)

  • 架构师筑基必备技能
  • Android高级UI与FrameWork源码
  • 360°全方面性能调优
  • 解读开源框架设计思想
  • NDK模块开发
  • 微信小程序
  • Hybrid 开发与Flutter

知识梳理完之后,就需要进行查漏补缺,所以针对这些知识点,我手头上也准备了不少的电子书和笔记,这些笔记将各个知识点进行了完美的总结:

Android开发七大模块核心知识笔记

《960全网最全Android开发笔记》

《379页Android开发面试宝典》

历时半年,我们整理了这份市面上最全面的安卓面试题解析大全
包含了腾讯、百度、小米、阿里、乐视、美团、58、猎豹、360、新浪、搜狐等一线互联网公司面试被问到的题目。熟悉本文中列出的知识点会大大增加通过前两轮技术面试的几率。

如何使用它?

1.可以通过目录索引直接翻看需要的知识点,查漏补缺。
2.五角星数表示面试问到的频率,代表重要推荐指数

《507页Android开发相关源码解析》

只要是程序员,不管是Java还是Android,如果不去阅读源码,只看API文档,那就只是停留于皮毛,这对我们知识体系的建立和完备以及实战技术的提升都是不利的。

真正最能锻炼能力的便是直接去阅读源码,不仅限于阅读各大系统源码,还包括各种优秀的开源库。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

1.可以通过目录索引直接翻看需要的知识点,查漏补缺。
2.五角星数表示面试问到的频率,代表重要推荐指数

[外链图片转存中…(img-4X8L3Xr5-1714762934752)]

《507页Android开发相关源码解析》

只要是程序员,不管是Java还是Android,如果不去阅读源码,只看API文档,那就只是停留于皮毛,这对我们知识体系的建立和完备以及实战技术的提升都是不利的。

真正最能锻炼能力的便是直接去阅读源码,不仅限于阅读各大系统源码,还包括各种优秀的开源库。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 15
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值