BF算法和KMP算法
这两天复习数据结构(严蔚敏版),记录第四章串中的两个重要算法,BF算法和KMP算法,博主主要学习Java,所以分析采用Java语言,后面会补上C语言的实现过程。
1、Brute-Force算法(暴力法)
要求:将主串的第i个字符(一般情况i为1)和字串的第一个字符进行比较。若相等,则继续比较后续字符;若不相等,则从主串的下一个字符起,重新与子串的第一个字符比较。成功,返回主串中与子串相匹配的子序列的第一个字符的序号;失败,返回0
public class Brute_Force {
/**
* 暴力法 ,O(m*n) (m、n分别为主串长度和子串长度)
* @param S 主串(原始字符串)
* @param T 子串(模式字符串)
* @return 如果模式字符串在原始字符串中存在,返回模式字符串在原始字符串中第一次出现的索引
*/
private int Index_BF(String S, String T) {
//习惯性排除错误的模式
if (S==null || S.length()<=0 || T==null || T.length()<=0)
return 0;
int S_length = S.length();
int T_length = T.length();
int i=0, j=0;
while (i<S_length && j<T_length) {
if (S.charAt(i) == T.charAt(j)) {
i++;
j++;
}else {
//不等
i = i-j+1; //这一步的思想是主串返回到的步数为子串移动的步数i-j,然后从下一个开始+1
j = 0; //j返回索引为0的位置
}
}
if (j == T_length) {
return i-T_length;
}
return 0;
}
//主方法,测试
public static void main(String[] args) {
Brute_Force bf = new Brute_Force();
String S = "hello world";
String T = "world";
System.out.println(bf.Index_BF(S,T)); //6
}
}
2、KMP算法
要求效果同上,只是更合理地应用部分已经匹配的结果
public class KMP {
/**
* KMP 高效法 ,O(m+n) (m、n分别为主串长度和子串长度)
* @param S 主串(原始字符串)
* @param T 子串(模式字符串)
* @return 如果模式字符串在原始字符串中存在,返回模式字符串在原始字符串中第一次出现的索引
*/
private int Index_KMP(String S, String T) {
//习惯性排除错误的模式
if (S==null || S.length()<=0 || T==null || T.length()<=0)
return 0;
//得到子串的next数组
int[] nextArr = getNextArray(T);
int S_length = S.length();
int T_length = T.length();
int i=0, j=0;
while (i<S_length && j<T_length) {
if (j==-1 || S.charAt(i) == T.charAt(j)) {//j==-1第一个字符就和当前测试的字符不相等
i++;
j++;
}else {
j = nextArr[j];
}
}
if (j==T_length) {
return i-T_length;
}
return 0;
}
/**
* 获取模式字符串的next数组
* @param str 子串
* @return
*/
private int[] getNextArray(String str) {
//习惯性排除错误的模式
if (str==null || str.length()<=0)
return null;
int length = str.length();
int[] nextArr = new int[length];
int j=0, k=-1;
nextArr[0] = -1;
while (j<length-1) {
if (k==-1 || str.charAt(j) == str.charAt(k)) {
j++;
k++;
nextArr[j] = k;
}else {
k = nextArr[k];
}
}
return nextArr;
}
//主方法,测试
public static void main(String[] args) {
KMP kmp = new KMP();
String S = "hello world";
String T = "world";
System.out.println(kmp.Index_KMP(S,T)); //6
}
}
3、KMP的讲解
KMP算法的核心,通俗来讲是,当S[i]和T[j]发生不匹配现象时,i指针不需要回溯,只需j指针回溯即可。详情可以看下这篇知乎:
https://www.zhihu.com/question/21923021/answer/281346746
大体的对比:
4、C语言实现
考试还是要用C语言,还是得好好参考下
BF算法:
#include <stdio.h>
#include <string.h>
//串普通模式匹配算法的实现函数,其中 B是主串,A是子串
int BF(char * B,char *A){
int i=0,j=0;
while (i<strlen(B) && j<strlen(A)) {
if (B[i]==A[j]) {
i++;
j++;
}else{
i=i-j+1;
j=0;
}
}
//跳出循环有两种可能,i=strlen(B)说明已经遍历完主串,匹配失败;j=strlen(A),说明子串遍历完成,在主串中成功匹配
if (j==strlen(A)) {
return i-strlen(A)+1;
}
//运行到此,为i==strlen(B)的情况
return 0;
}
int main() {
int number = BF("hello world", "world");
printf("%d",number);
return 0;
}
KMP算法:
#include <stdio.h>
#include <string.h>
void Next(char*T,int *next){
int i=1;
next[1]=0;
int j=0;
while (i<strlen(T)) {
if (j==0||T[i-1]==T[j-1]) {
i++;
j++;
next[i]=j;
}else{
j=next[j];
}
}
}
int KMP(char * S,char * T){
int next[10];
Next(T,next);//根据模式串T,初始化next数组
int i=1;
int j=1;
while (i<=strlen(S)&&j<=strlen(T)) {
//j==0:代表模式串的第一个字符就和当前测试的字符不相等;S[i-1]==T[j-1],如果对应位置字符相等,两种情况下,指向当前测试的两个指针下标i和j都向后移
if (j==0 || S[i-1]==T[j-1]) {
i++;
j++;
}
else{
j=next[j];//如果测试的两个字符不相等,i不动,j变为当前测试字符串的next值
}
}
if (j>strlen(T)) {//如果条件为真,说明匹配成功
return i-(int)strlen(T);
}
return -1;
}
int main() {
int i=KMP("hello world","world");
printf("%d",i);
return 0;
}
BF算法和KMP算法的更多相关文章
- 字符串匹配-BF算法和KMP算法
声明:图片及内容基于https://www.bilibili.com/video/av95949609 BF算法 原理分析 Brute Force 暴力算法 用来在主串中查找模式串是否存以及出现位置 ...
- 字符串匹配的BF算法和KMP算法学习
引言:关于字符串 字符串(string):是由0或多个字符组成的有限序列.一般写作`s = "123456..."`.s这里是主串,其中的一部分就是子串. 其实,对于字符串大小关系 ...
- 串的模式匹配 BF算法和KMP算法
设有主串s和子串t,子串t的定位就是要在主串中找到一个与子串t相等的子串.通常把主串s称为目标串,把子串t称为模式串,因此定位也称为模式匹配. 模式匹配成功是指在目标串s中找到一个模式串t: 不成功则 ...
- 串匹配模式中的BF算法和KMP算法
考研的专业课以及找工作的笔试题,对于串匹配模式都会有一定的考察,写这篇博客的目的在于进行知识的回顾与复习,方便遇见类似的题目不会纠结太多. 传统的BF算法 传统算法讲的是串与串依次一对一的比较,举例设 ...
- BF算法和KMP算法 python实现
BF算法 def Index(s1,s2,pos = 0): """ BF算法 """ i = pos j = 0 while(i < ...
- 字符串匹配(BF算法和KMP算法及改进KMP算法)
#include <stdio.h> #include <string.h> #include <stdlib.h> #include<cstring> ...
- BF算法和KMP算法(javascript版本)
var str="abcbababcbababcbababcabcbaba";//主串 var ts="bcabcbaba";//子串 function BF( ...
- 数据结构(十六)模式匹配算法--Brute Force算法和KMP算法
一.模式匹配 串的查找定位操作(也称为串的模式匹配操作)指的是在当前串(主串)中寻找子串(模式串)的过程.若在主串中找到了一个和模式串相同的子串,则查找成功:若在主串中找不到与模式串相同的子串,则查找 ...
- 软件设计师_朴素模式匹配算法和KMP算法
1.从主字符串中匹配模式字符串(暴力匹配) 2. KMP算法
随机推荐
- Lua io.lines()
前言# 从文章的题目可以看出,今天的内容是和文件的行相关的,其实这个函可以看成是一个文件读取函数,只不过文件读取的形式固定了,就是只能一行一行的读,接下来我们就一起来看看这个函数究竟要怎么使用. 内容 ...
- 【PHP数据结构】图的应用:最短路径
上篇文章的最小生成树有没有意犹未尽的感觉呀?不知道大家掌握得怎么样,是不是搞清楚了普里姆和克鲁斯卡尔这两种算法的原理了呢?面试的时候如果你写不出,至少得说出个大概来吧,当然,如果你是要考研的学生,那就 ...
- 让PHP能够调用C的函数-FFI扩展
在大型公司中,一般会有很我编程语言的配合.比如说让 Java 来做微服务层,用 C++ 来进行底层运算,用 PHP 来做中间层,最后使用 JS 展现效果.这些语言间的配合大部分都是通过 RPC 来完成 ...
- Kotlin协程入门
开发环境 IntelliJ IDEA 2021.2.2 (Community Edition) Kotlin: 212-1.5.10-release-IJ5284.40 介绍Kotlin中的协程.用一 ...
- css布局宽度自适应
随着各种终端的不断涌现,网页中的元素适应不同的分辨率变得特别重要,根据经验,涉及到宽度自适应的一共有四种情况: 左端固定,右边自适应:右端固定,左边自适应:两端固定,中间自适应:中间固定,两端自适应. ...
- 没想到 TCP 协议,还有这样的骚操作。。。
大家好,我是小林. 昨晚有位读者问了我这么个问题: 大概意思是,一个已经建立的 TCP 连接,客户端中途宕机了,而服务端此时也没有数据要发送,一直处于 establish 状态,客户端恢复后,向服务端 ...
- P7518-[省选联考2021A/B卷]宝石【主席树,二分】
正题 题目链接:https://www.luogu.com.cn/problem/P7518 题目大意 给出\(n\)个点的一棵树,每个点上有不大于\(m\)的数字. 然后给出一个长度为\(c\)的各 ...
- 项目配置shiro原缓存注解失效
项目用springboot + shiro + ehcache @cacheable 注解不起作用原因 Shiro框架初始化比Spring框架的某些部件早,导致使用@Autowire注入Shiro框架 ...
- 深度学习--GAN学习笔记
生成模型 WGAN Blog GAN 推荐学习网站 生成模型 什么是生成模型? GMM: 用来做聚类,(非监督学习) NB(朴素贝叶斯):(监督学习,可以用来做垃圾邮件分类) Logistics 回归 ...
- 100台机器上海量IP如何查找出现频率 Top 100?
场景题 有 100 机器,每个机器的磁盘特别大,磁盘大小为 1T,但是内存大小只有 4G,现在每台机器上都产生了很多 ip 日志文件,每个文件假设有50G,那么如果计算出这 100 太机器上访问量最多 ...