字符串---KMP

作者: 哟破赛呦 | 来源:发表于2019-01-17 09:40 被阅读13次

求模式串在目标串中出现的次数和位置

next数组的一些性质

KMP最小循环节、循环周期:
定理:假设S的长度为len则S存在最小循环节,对S构造next数组,循环节的长度Llen-next[len],子串为S[0…len-next[len]-1]。
(1)如果len可以被len - next[len]整除,则表明字符串S可以完全由循环节循环组成,循环周期T = len/L
(2)如果不能,说明还需要再添加几个字母才能补全。需要补的个数是循环个数L-len%L = L-(len-L)%L = L-next[len]%LL = len-next[len]

代码

namespace KMP{
    int next[10010];
    // 处理模式串的next数组,x[i-next[i]...i-1] == x[0....next[i]-1]
    void kmp_pre(char x[], int m, int _next[]){
        int i, j;
        j = _next[0] = -1;
        i = 0;
        while(i < m){
            while(-1 != j && x[i] != x[j]) j = _next[j];
            _next[++i] = ++j;
        }
    }
    // 改进next数组,减少回溯次数,但不能使用next的性质
    void kmp_pre_2(char x[], int m, int fast_next[]){
        int i, j;
        j = fast_next[0] = -1;
        i = 0;
        while(i < m){
            while(-1 != j && x[i] != x[j]) j = fast_next[j];
            if(x[++i] == x[++j]) fast_next[i] = fast_next[j];
            else fast_next[i] = j;
        }
    }
    //模式匹配,返回出现次数,x模式串,y主串
    int kmp(char x[], int m, char y[], int n){
        int i, j;
        int ans = 0;
        kmp_pre(x, m, next);
        //kmp_pre_2(x, m, next);
        i = j = 0;
        while(i < n){
            while(-1 !=j && y[i] != x[j]) j = next[j];
            i++; j++;
            if(j >= m){
                ans++;
                //i-m即为模式串在主串中的开始位置
                j = next[j];
            }
        }
        return ans;
    }
}

例题

hdu3336 next数组+dp
大意:给你个字符串如abab,它的子串aababaabab出现次数
即2+2+1+1=6
dp[i]表示以第i个字符结尾的子串出现次数
dp[i] = dp[next[i]] + 1
对于abab,next[]={-1,0,0,1,2}
当i=1时,表示子串a,dp[1] = dp[next[1]] + 1 = dp[0] + 1 = 1,‘a’
当i=2时,表示子串ab,dp[2] = dp[next[2]] + 1 = dp[0] + 1 = 1,'ab'
当i=3时,表示子串aba,dp[3] = dp[next[3]] + 1 = dp[1] + 1 = 2,'a','aba'
当i=4时,表示子串abab,dp[4] = dp[next[4]] + 1 = dp[2] + 1 = 2,'ab','abab'
以a结尾的有'a','a','aba';
以a结尾的有'ab','ab','abab';

#include <stdio.h>
#include <string.h>
#define mem(x,y) memset(x,y,sizeof(x))
const int mod = 10007;
namespace KMP{
    int next[200001];
    // 处理模式串的next数组,x[i-next[i]...i-1] == x[0....next[i]-1]
    void kmp_pre(char x[], int m, int _next[]){
        int i, j;
        j = _next[0] = -1;
        i = 0;
        while(i < m){
            while(-1 != j && x[i] != x[j]) j = _next[j];
            _next[++i] = ++j;
        }
    }
}
int t,n;//200000
char str[200001];
int dp[200001];
int main(){
    scanf("%d",&t);
    while(t--){
        mem(KMP::next,0);
        scanf("%d",&n);
        scanf("%s",str);
        KMP::kmp_pre(str,n,KMP::next);
        int ans=0;
        mem(dp,0);
        for(int i=1;i<=n;i++){
            dp[i] = (dp[KMP::next[i]]+1)%mod;
            ans = (ans + dp[i])%mod;
        }
        printf("%d\n",ans );
    }
    return 0;
}

相关文章

  • KMP算法文章合集

    字符串的查找:朴素查找算法和KMP算法 暴力匹配算法与KMP算法(串的匹配) 字符串查找算法BF和KMP 字符串匹...

  • KMP字符串匹配

    KMP字符串匹配

  • 算法(2)KMP算法

    1.0 问题描述 实现KMP算法查找字符串。 2.0 问题分析 “KMP算法”是对字符串查找“简单算法”的优化。 ...

  • KMP

    KMP有什么用 KMP主要应用在字符串匹配上。 KMP的主要思想是「当出现字符串不匹配时,可以知道一部分之前已经匹...

  • KMP算法的JS实现

    talk is cheap,show me the code: 参考链接: 阮一峰-字符串匹配的KMP算法[KMP...

  • JavaScript 二分查找 & KMP 算法

    KMP 查找 Knuth-Morris-Pratt字符串查找算法(简称为KMP算法)可在一个主文本字符串 str1...

  • KMP算法(字符串匹配问题)

    一、是什么? 注意,是KMP算法,不是MMP哈,我没有骂人。KMP算法是用来做字符串匹配的,除了KMP算法分,还有...

  • 分享几道常见字符串算法题

    1. KMP 算法 谈到字符串问题,不得不提的就是 KMP 算法,它是用来解决字符串查找的问题,可以在一个字符串(...

  • KMP算法理解

    文章大纲:1.KMP算法概念2.KMP算法中最核心的next[] 数组是如何生成的3.使用KMP算法 匹配字符串 ...

  • kmp 算法

    前言: 少 kmp 多学习 0X00 算法板子 831. KMP字符串 感觉还是很难理解

网友评论

    本文标题:字符串---KMP

    本文链接:https://www.haomeiwen.com/subject/fwsbdqtx.html