KMP模板题

KMP模板:

#include<iostream>
using namespace std;

const int MAX_S=1000005;
int n,m;
string str,st;
int Next[MAX_S];

void GetNext();
int KMP();
int main()
{
	ios::sync_with_stdio(false);
	while(cin>>str>>st){
		n=str.size();	m=st.size();
		cout<<KMP()<<endl; 
	}
	
	return 0;
}

void GetNext()
{
	int k=-1;
	Next[0]=-1;
	for(int i=1;i<m;++i)
	{
		while(k!=-1&&st[k+1]!=st[i]){
			k=Next[k];	//下一匹配位置 
		}
		if(st[k+1]==st[i])	++k;
		Next[i]=k;
	}
}

int KMP()
{
	int ans=-1;
	GetNext();
	for(int i=0,j=-1;i<n;++i)
	{
		while(j!=-1&&st[j+1]!=str[i]){
			j=Next[j];
		}
		if(st[j+1]==str[i])	++j;
		if(j==m-1){
			// j=-1;//重新寻找匹配,匹配字符串之间不重叠,若匹配字符串之间可重叠则不变 
			ans=i-j;	break;
		}
	}
	//返回匹配首下标,为-1表示匹配失败 
	return ans;
}

 

一,求匹配的首下标  HDU-1711-Number Sequence  直接套用模板。。

Code :

#include<iostream>
using namespace std;

const int MAX_S=1000005;
const int MAX_M=10005;
int n,m,T;
int str[MAX_S],st[MAX_M];
int Next[MAX_M];

void GetNext();
int KMP();
int main()
{
	ios::sync_with_stdio(false);
	cin>>T;
	while(T--){
		cin>>n>>m;
		for(int i=0;i<n;++i)
			cin>>str[i];
		for(int i=0;i<m;++i)
			cin>>st[i];
		cout<<KMP()<<endl;
	}
	
	return 0;
}

void GetNext()
{
	int k=-1;
	Next[0]=-1;
	for(int i=1;i<m;++i)
	{
		while(k!=-1&&st[k+1]!=st[i]){
			k=Next[k];	//下一匹配位置 
		}
		if(st[k+1]==st[i])	++k;
		Next[i]=k;
	}
}

int KMP()
{
	int ans=-1;
	GetNext();
	for(int i=0,j=-1;i<n;++i)
	{
		while(j!=-1&&st[j+1]!=str[i]){
			j=Next[j];
		}
		if(st[j+1]==str[i])	++j;
		if(j==m-1){
			ans=i-j+1;	break;
		}
	}
	return ans;
}

 

二,求匹配次数,匹配字符串之间可重叠 HDU-1686-Oulipo  查找到直接计数即可

Code :

#include<iostream>
using namespace std;

const int MAX_S=10005;
int n,m,T;
string str,st;
int Next[MAX_S];

void GetNext();
int KMP();
int main()
{
	ios::sync_with_stdio(false);
	cin>>T; 
	while(T--){
		cin>>st>>str;
		n=str.size();	m=st.size();
		cout<<KMP()<<endl;
	}
	
	return 0;
}

void GetNext()
{
	int k=-1;
	Next[0]=-1;
	for(int i=1;i<m;++i)
	{
		while(k!=-1&&st[k+1]!=st[i]){
			k=Next[k];
		}
		if(st[k+1]==st[i])	++k;
		Next[i]=k;
	}
}

int KMP()
{
	int ans=0;
	GetNext();
	for(int i=0,j=-1;i<n;++i)
	{
		while(j!=-1&&st[j+1]!=str[i]){
			j=Next[j];
		}
		if(st[j+1]==str[i])	++j;
		if(j==m-1)	++ans;
	}
	
	return ans;
}

 

三,求匹配次数,匹配字符串之间不可重叠 HDU-2087-剪花布条  查找到后将 j=-1即可

Code :

#include<iostream>
using namespace std;

const int MAX_S=10005;
int n,m,T;
string str,st;
int Next[MAX_S];

void GetNext();
int KMP();
int main
{
	ios::sync_with_stdio(false);
	while(cin>>str&&str!="#"){
		cin>>st;
		n=str.size();	m=st.size();
		cout<<KMP()<<endl;
	}
	
	return 0;
}

void GetNext()
{
	int k=-1;
	Next[0]=-1;
	for(int i=1;i<m;++i)
	{
		while(k!=-1&&st[k+1]!=st[i]){
			k=Next[k];
		}
		if(st[k+1]==st[i])	++k;
		Next[i]=k;
	}
}

int KMP()
{
	int ans=0;
	GetNext();
	for(int i=0,j=-1;i<n;++i)
	{
		while(j!=-1&&st[j+1]!=str[i]){
			j=Next[j];
		}
		if(st[j+1]==str[i])	++j;
		if(j==m-1){
			j=-1;	++ans;
		}
	}
	
	return ans;
}

这些只是最基础的KMP模板题,想更深去学习的可以去这篇博客学习 KMP模板以及简单的入门题总结 

### C++ 算法题目模板代码示例 #### 排序算法 对于排序问题,可以采用多种方法实现。以下是冒泡排序的一个简单例子: ```cpp void BubbleSort(int a[], int n) { for (int i = 0; i < n - 1; i++) { for (int j = 0; j < n - i - 1; j++) { if (a[j] > a[j + 1]) { int t = a[j]; a[j] = a[j + 1]; a[j + 1] = t; } } } } ``` 此段代码实现了基本的冒泡排序逻辑[^1]。 #### 字符串匹配算法 当涉及到字符串匹配时,暴力求解是一种直观的方法。下面是一个简单的暴力匹配程序片段: ```cpp #include <iostream> using namespace std; int main() { int n, m; string s, p; // s为较长字符串(模式串) cin >> n >> p >> m >> s; for (int i = 0; i < m; i++) { bool flag = true; // 最开始是匹配成功的状态 for (int j = 0; j < n; j++) { if (s[i + j] != p[j]) { flag = false; // 只要有一个不通过,则设置为false并跳出循环 break; } } if (flag == true) { cout << i << " "; } } return 0; } ``` 这段代码展示了如何利用双重循环来逐字符比较两个字符串,并输出所有可能的位置索引[^2]。 #### KMP算法中的next数组处理部分 在更高效的KMP算法中,`next[]` 数组用于记录前缀函数值,从而加速模式串与目标串之间的匹配过程。这里给出了一段更新 `j` 的关键语句: ```cpp while(j >= 0 && s[i] != t[j + 1]) { j = next[j]; } ``` 该行代码的作用是在发生失配的情况下调整指针位置以便继续尝试其他潜在匹配点[^3]。 #### 快速幂运算及其应用——费马小定理求逆元 快速幂运算是计算大数次方的有效手段之一,在模意义下尤其有用。下面是基于二分思想优化后的快速幂函数定义以及其应用于求取乘法逆元的具体实例: ```cpp // 计算(a^b)%c的结果 int qpow(int a, int b, int c){ int res = 1; while(b){ if(b & 1) res = res * a % c; b >>= 1; a = a * a % c; } return res; } // 使用费马小定理求得a关于mod的乘法逆元 inline long long inv(long long base,long long mod){ return qpow(base,mod-2,mod); } ``` 上述代码不仅提供了快速幂的基础框架,还演示了怎样借助于费马小定理高效地找到特定条件下存在的乘法逆元[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值