1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
| //主逻辑,重点在于next的求法,看 nextArr 方法 //i不动j动 function kmpSearch(string $s, string $p) { $sLen = strlen($s); $pLen = strlen($p); $i = 0; $j = 0; $next = nextArr($p); while ($i < $sLen && $j < $pLen) { if ($j == -1 || $s[$i] == $p[$j]) { $i++; $j++; }else{ $j=$next[$j]; } } if ($j == $pLen){ return $i - $j; } return -1; }
//返回 next 数组,即是每个子串的最大公共长度数组右移一位,然后填充 -1 //也可以理解成这个字符之前的字符串中有多大长度的相同前缀后缀 function nextArr(string $p): array { $res = []; $res[0] = -1; $pLen = strlen($p); for ($i = 1; $i < $pLen; $i++) { $res[$i] = maxLen(mb_substr($p, 0, $i)); } return $res; } //返回字符串前后缀公共长度 function maxLen($str): int { $maxLen = 0; $sLen = strlen($str); if ($sLen != 1) { for ($i = 2; $i <= $sLen; $i++) { if (mb_substr($str, 0, $i - 1) == mb_substr($str, $sLen - $i + 1, $i - 1)) { $maxLen = $i - 1; } } } return $maxLen; }
echo kmpSearch('bbc abcdab abcdabcdabde','abcdabd');
|