網站首頁 編程語言 正文
C語言實現KMP
- 先看代碼
- 代碼解讀
- 歸根結底就是不要走重復的路
- 先看部分匹配表的核心代碼
先看代碼
#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;
}
//獲取一個匹配字符串的部分匹配表
int* kmpNext(char* myStr){
//創建一個數組保存部分匹配值
int* arr = (int*)malloc(sizeof(int)* strlen(myStr));
arr[0]=0; //如果字符串長度是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;
}
代碼解讀
歸根結底就是不要走重復的路
先要根據目標匹配字符串構建部分匹配表
先看部分匹配表的核心代碼
int* arr = (int*)malloc(sizeof(int)* strlen(myStr));
arr[0]=0; //如果字符串長度是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;
}
解讀:當目標字符串就一個字符的時候,部分匹配表就是0,沒啥說的。當目標字符串長度超過1的時候,讓j=0,i=1,這樣就不斷的比較不斷加長的前后綴。(尋找最長就行)
例如:j=0,i=1;就是AB,兩個字符不相等;arr[1]=0;-----------,類推
j=0,i=4;就是ABCDA,此時兩個字符相等,arr[4]=1,注意此時j=1;
j=1,i=5,就是ABCDAB,此時第二個字符也相等,也就是前后綴有共同的AB長度的子串,arr[i]=2,j=2;
j=2,i=6,就是ABCDABD,此時第三個字符不相等,此時執行while邏輯,j=2>0; j=arr[2-1]=0;
最后就構成了arr[]={0,0,0,0,1,2,0};的部分匹配表。
其中最難理解是這行代碼:
j = arr[j-1]; //發現不匹配,就需要返回上一次的最長的公共前后綴的前綴起始位置???
可以以AADAAA這個字符串,來進行理解。就是當D和A不相等的時候,讓最長公共前綴退后一位,再去比較。這個時候AA和AA又成為了前后綴的公共前綴。
原文鏈接:https://blog.csdn.net/Smallcloudy/article/details/126089165
- 上一篇:linux交叉編譯依賴包
- 下一篇:C語言排序算法實現
相關推薦
- 2022-10-14 laravel 中關于模型查詢構造器的特殊用法
- 2022-10-28 Django靜態文件配置request對象方法ORM操作講解_python
- 2023-03-21 Android?Hilt依賴注入的實現淺析_Android
- 2022-06-07 Docker的四種網絡模式_docker
- 2022-03-03 iview 在 Table 組件中,文字過長用省略號代替,鼠標放上去 Tooltip 文字提示
- 2022-05-15 Python獲取網絡圖片和視頻的示例代碼_python
- 2022-07-19 簡單認清深拷貝和淺拷貝
- 2023-04-06 C#中的那些警告該如何去除(完全去除C#警告)_C#教程
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支