網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
C語(yǔ)言實(shí)現(xiàn)KMP
- 先看代碼
- 代碼解讀
- 歸根結(jié)底就是不要走重復(fù)的路
- 先看部分匹配表的核心代碼
先看代碼
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
/**
*
* @param srcStr
* @param descStr
* @param arr 部分匹配表
* @return
*/
int kmpSearch(char* srcStr,char* descStr,int* arr){
for (int i = 0,j=0; i < strlen(srcStr); ++i) {
while (j>0 && srcStr[i] != descStr[j]){
j=arr[j-1];
}
if(srcStr[i] == descStr[j]){
j++;
}
if(j== strlen(descStr)){
return i-j+1;
}
}
return -1;
}
//獲取一個(gè)匹配字符串的部分匹配表
int* kmpNext(char* myStr){
//創(chuàng)建一個(gè)數(shù)組保存部分匹配值
int* arr = (int*)malloc(sizeof(int)* strlen(myStr));
arr[0]=0; //如果字符串長(zhǎng)度是1,部分匹配值就是0
for (int i = 1,j=0; i < strlen(myStr) ; ++i) {
while (j>0 && myStr[i]!= myStr[j]){
j = arr[j-1];
}
if(myStr[i] == myStr[j]){
j++;
}
arr[i]=j;
}
return arr;
}
int main() {
char* srcStr = "hello world";
char* descStr = "wo";
int* arr = kmpNext(descStr);
int index = kmpSearch(srcStr,descStr,arr);
printf("%d",index);
return 0;
}
代碼解讀
歸根結(jié)底就是不要走重復(fù)的路
先要根據(jù)目標(biāo)匹配字符串構(gòu)建部分匹配表
先看部分匹配表的核心代碼
int* arr = (int*)malloc(sizeof(int)* strlen(myStr));
arr[0]=0; //如果字符串長(zhǎng)度是1,部分匹配值就是0
for (int i = 1,j=0; i < strlen(myStr) ; ++i) {
while (j>0 && myStr[i]!= myStr[j]){
j = arr[j-1]; //發(fā)現(xiàn)不匹配,就需要返回上一次的最長(zhǎng)的公共前后綴的前綴起始位置???
}
if(myStr[i] == myStr[j]){
j++;
}
arr[i]=j;
}
解讀:當(dāng)目標(biāo)字符串就一個(gè)字符的時(shí)候,部分匹配表就是0,沒啥說的。當(dāng)目標(biāo)字符串長(zhǎng)度超過1的時(shí)候,讓j=0,i=1,這樣就不斷的比較不斷加長(zhǎng)的前后綴。(尋找最長(zhǎng)就行)
例如:j=0,i=1;就是AB,兩個(gè)字符不相等;arr[1]=0;-----------,類推
j=0,i=4;就是ABCDA,此時(shí)兩個(gè)字符相等,arr[4]=1,注意此時(shí)j=1;
j=1,i=5,就是ABCDAB,此時(shí)第二個(gè)字符也相等,也就是前后綴有共同的AB長(zhǎng)度的子串,arr[i]=2,j=2;
j=2,i=6,就是ABCDABD,此時(shí)第三個(gè)字符不相等,此時(shí)執(zhí)行while邏輯,j=2>0; j=arr[2-1]=0;
最后就構(gòu)成了arr[]={0,0,0,0,1,2,0};的部分匹配表。
其中最難理解是這行代碼:
j = arr[j-1]; //發(fā)現(xiàn)不匹配,就需要返回上一次的最長(zhǎng)的公共前后綴的前綴起始位置???
可以以AADAAA這個(gè)字符串,來進(jìn)行理解。就是當(dāng)D和A不相等的時(shí)候,讓最長(zhǎng)公共前綴退后一位,再去比較。這個(gè)時(shí)候AA和AA又成為了前后綴的公共前綴。
原文鏈接:https://blog.csdn.net/Smallcloudy/article/details/126089165
- 上一篇:linux交叉編譯依賴包
- 下一篇:C語(yǔ)言排序算法實(shí)現(xiàn)
相關(guān)推薦
- 2022-07-21 基于Spring Boot項(xiàng)目構(gòu)建流水線
- 2022-04-16 pycharm三個(gè)有引號(hào)不能自動(dòng)生成函數(shù)注釋的問題_python
- 2022-08-15 element的form表單中如何一行顯示多el-form-item標(biāo)簽
- 2023-01-02 Kotlin?fun函數(shù)使用方法_Android
- 2022-04-15 C#的并發(fā)機(jī)制優(yōu)秀在哪你知道么_C#教程
- 2022-07-07 redis連接報(bào)錯(cuò)error:NOAUTH?Authentication?required_Redi
- 2023-12-11 Mybatis結(jié)果集映射ResultMap
- 2022-12-29 Redis并發(fā)訪問問題詳細(xì)講解_Redis
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支