#include <bits/stdc++.h>
using namespace std;
const int N=1e6+100;
char T[2*N],S[N];
int nex[N];
bool equ(char c1,char c2)//字符串匹配函数
{
if(c2-32==c1||c1-32==c2)return true;//同一个字母的大小写,匹配成功
else return false;//否则匹配失败
}
void get_next(int length)//求next数组
{
nex[0]=nex[1]=0;
for(int i=2,j=0;i<=length;i++)
{
while(j&&(S[i]!=S[j+1]))j=nex[j];
if(S[i]==S[j+1])j++;
nex[i]=j;
}
}
int KMP(int T_length,int S_length)//在文本串T中匹配模式串S
{
for(int i=1,j=0;i<=T_length;i++)
{
while(j&&(!equ(T[i],S[j+1])))j=nex[j];
if(equ(T[i],S[j+1]))j++;
if(j==S_length)//匹配成功
{
return i-S_length;//返回首个匹配成功的位置
}
}
return -1;//匹配失败,返回-1
}
int main()
{
int n;
cin>>n;
cin>>T+1;
cin>>S+1;
for(int i=1;i<=n;i++)T[i+n]=T[i];//将文本串扩大一倍,实现环变链
get_next(n);//对模式串S求其next数组
int ans=KMP(2*n,n);//在扩大一倍后的文本串T中匹配模式串S
if(ans!=-1)//匹配成功
{
cout<<"Yes"<<endl;
cout<<min(ans,n-ans)<<endl;//顺时针和逆时针两种方案,取较小者
//如n=5,求得ans=4,需逆时针移动4位,则转换为顺时针移动5-4=1位
}
else cout<<"No"<<endl;//匹配失败
return 0;
}
契合匹配 kmp 破环成链
最新推荐文章于 2024-07-24 22:13:06 发布