暴力匹配算法(BF)与KMP算法分别实现字符串的模式匹配

设有两个字符串s和t,首先将s1与t1进行比较,直到s的某一个字符si和ti相同,再将它们之后的字符进行比较,若也相同,则如此继续往下比较,当s的某一个字符si与t的字符tj不同时,则s返回到本趟开始字符的下一个字符,即si-j+2,t返回到t1,继续开始下一趟的比较,重复上述过程。若t中的字符全部比较完,则说明本趟匹配成功,本趟的起始位置是i-j+1,否则,匹配失败。

#include<stdio.h>
#include<iostream>
#include<string>

int  BF_Index(std::string S, std::string T, int pos);//暴力匹配算法

int KMP_Index(std::string S, std::string T, int pos);//KMP算法

void getNext(std::string S, int  next[]);//获取next值

static std::string getCorrectString() {
	bool psw = false;                  //循环标志位
	std::string intput, str;
	do {                                     //对于空格加以判断,防止读入错误
		psw = false;
		std::cin >> intput;
		str = str + intput;
		while (std::cin.peek() == ' ') { str = str + " "; std::cin.get(); psw = true; }
	} while (psw);
	return str;
}

int main() {
	std::string  SString, TString;
	int pos;
	std::cout << "请输入主字符串:";
	bool psw = false;                  //循环标志位

	SString = getCorrectString();                                 //对于空格加以判断,防止读入错误
	std::cout << std::endl;
	std::cout << "请输入搜索串:";
	TString = getCorrectString();                            //对于空格加以判断,防止读入错误
	std::cout << std::endl;
	std::cout << "请输入要检索的开始位置:";
	std::cin >> pos;
	std::cout << std::endl;

	if (BF_Index(SString, TString, pos) != 0 && KMP_Index(SString, TString, pos) != 0) {
		std::cout << "BF匹配成功开始位置是:第  " << BF_Index(SString, TString, pos) << "  个位置" << std::endl;
		std::cout << "KMP匹配成功开始位置是:第  " << KMP_Index(SString, TString, pos) << "  个位置" << std::endl;
	}
	else
	{
		std::cout << "BF匹配失败" << std::endl;
		std::cout << "KMP匹配失败" << std::endl;
	}
	return 0;

}

int  BF_Index(std::string S, std::string T, int pos) {
	int Slength = S.length();     //获取字符串长度
	int Tlength = T.length();
	int i = pos - 1, j = 0;
	while (i < Slength && j < Tlength)    //各自下表小于对应字符串长度
	{
		if (S[i] == T[j]) {    //相等则判断下一位是否相等
			i++;
			j++;
		}
		else        //匹配失败,从头再来
		{
			i = i - j + 1;
			j = 0;
		}
	}
	if (j == Tlength)return  i - Tlength + 1;     //若最后后一位也匹配,则++,于是j==Tlength
	else
	{
		return 0;                 //检索失败返回0
	}
}
int KMP_Index(std::string S, std::string T, int pos) {
	int Slength = S.length();     //获取字符串长度
	int Tlength = T.length();
	int i = pos - 1, j = 0;
	int* next = (int*)malloc(sizeof(int) * (Tlength + 1));
	getNext(T, next);
	while (i < Slength && j < Tlength)
	{
		if (S[i] == T[j] || j == 0) {    //若j==0;或者相等的话则判断下一位
			i++; j++;
		}
		else
		{
			j = next[j];            //否则为j=next[j]
		}
	}
	if (j == Tlength)return i - Tlength + 1;
	else
	{
		return 0;
	}
}

void getNext(std::string T, int  next[]) {//获取next数组
	int j = 1, k = 0;
	//next[0] = 0;
	next[1] = 0;
	while (j < T.length())
	{
		if (k == 0 || T[j] == T[k]) {   //主体思想是若相等则下一位的next要加一,否则一直往回走,直到k==0
			j++; k++; next[j] = k;
		}
		else {
			k = next[k];
		}
	}
}
  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值