转载:http://www.cnblogs.com/vongang/archive/2012/05/04/2483419.html

记一个KMP算法的应用,经典的KMP算法详解还是看这里

问题:给一个串,求这个串前i位构成的前缀由多少个子串组成。

比如aabaabaabaab,前2位是aa,a重复了2次,前6位是aabaab,aab重复了2次,前9位是aabaabaab,aab重复了3次,前12位是aabaabaabaab,aab重复了4次。

先说一下next()函数。pre[i] = j表示   S[1...j] = S[i - j....i];

下面讨论当i % (i - pre[i]) == 0 时,例如i = 12, pre[12] = 9:

如图。

S[1...9] == S[3...12];

因为已知 i % (i - pre[i]) == 0; 那么把i分成 i / (i - pre[i])段。

已知:

s3 == t3;

s2 == t2;

s1 == t1;

又因为t3 == s2, t1 == s1。所以 t1 = t2 = t3 = s1 = s2 = s3,也就是说 i / (i - pre[i])这几段中每一段都相等。

现在回到原问题:求这个串前i位构成的前缀由多少个子串组成,只需要找到i / (i - pre[i]) == 0的点i,则共有 i / (i - pre[i])个相同的子串构成前缀S[1...i]。

练习:POJ 1961, POJ 2406

附1961的代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <stack>
#include <cmath>
#include <algorithm> #define CL(arr, val) memset(arr, val, sizeof(arr))
#define REP(i, n) for(i = 0; i < n; ++i)
#define FOR(i, l, h) for(i = l; i <= h; ++i)
#define FORD(i, h, l) for(i = h; i >= l; --i)
#define L(x) x << 1
#define R(x) x << 1 | 1
#define MID(l, r) (l + r) >> 1
typedef long long LL; using namespace std; const int N = ; char s[N];
int pre[N];
int dp[N]; //这里加了一个数组,记录到i位置时所构成的前缀由多少个子串组成。
int n; void Next() {
int i, j = -;
pre[] = -;
for(i = ; i < n; ++i) {
while(j > - && s[j+] != s[i]) j = pre[j];
if(s[j+] == s[i]) ++j;
pre[i] = j;
}
} int main() {
freopen("data.in", "r", stdin); int i, cas = ;
while(scanf("%d", &n), n) {
scanf("%s", s);
printf("Test case #%d\n", ++cas);
Next();
REP(i, n + ) dp[i] = ;
FOR(i, , n - ) {
if((i + ) % (i - pre[i]) == && pre[i] != -) {
dp[i] = dp[pre[i]] + ; //到i的前缀就等于到pre[i]的前缀子串数加上 [pre[i], i]这个子串。
printf("%d %d\n", i + , dp[i]); //其实直接输出(i + 1)/(i - pre[i])就行,这里写搓了。。。T_T
}
}
cout << endl;
}
return ;
}

KMP算法Next()函数的一个应用的更多相关文章

  1. KMP算法-next函数求解

    KMP函数求解:一种改进的字符串匹配算法,由D.E.Knuth,J.H.Morris和V.R.Pratt同时发现,因此人们称它为KMP算法.KMP算法的关键是利用匹配失败后的信息,尽量减少模式串与主串 ...

  2. KMP算法的正确性证明及一个小优化

    直接把作业帖上来是不是有点不太公道呀... 无所谓啦反正各位看着开心就行 KMP算法 对于模式串$P$,建立其前缀函数$ N$ ,其中$N [q] $ 表示在$P$中,以$q$位置为结束的可以匹配到前 ...

  3. KMP算法解析(转自图灵社区)

    KMP算法是一个很精妙的字符串算法,个人认为这个算法十分符合编程美学:十分简洁,而又极难理解.笔者算法学的很烂,所以接触到这个算法的时候也是一头雾水,去网上看各种帖子,发现写着各种KMP算法详解的转载 ...

  4. 真正理解KMP算法

    作者:jostree 转载请注明出处 http://www.cnblogs.com/jostree/p/4403560.html 所谓KMP算法,就是判断一个模式串是否是一个字符串的子串,通常的算法当 ...

  5. 字符串匹配算法之 kmp算法 (python版)

    字符串匹配算法之 kmp算法 (python版) 1.什么是KMP算法 KMP是三位大牛:D.E.Knuth.J.H.MorriT和V.R.Pratt同时发现的.其中第一位就是<计算机程序设计艺 ...

  6. 串匹配算法讲解 -----BF、KMP算法

      参考文章: http://www.matrix67.com/blog/archives/115     KMP算法详解 http://blog.csdn.net/yaochunnian/artic ...

  7. BZOJ 1009 HNOI2008 GT考试 KMP算法+矩阵乘法

    标题效果:给定的长度m数字字符串s.求不包括子s长度n数字串的数目 n<=10^9 看这个O(n)它与 我们不认为这 令f[i][j]长度i号码的最后的字符串j位和s前者j数字匹配方案 例如,当 ...

  8. (原创)白话KMP算法详解

    引子:BF暴力算法 KMP算法知名度相当高,燃鹅其理解难度以及代码实现对于初学数据结构和算法的同学并不友好,经过两天的总结,详细总结KMP算法如下: 初学串的模式匹配时,我们都会接触到,或者说应该能想 ...

  9. 串的模式之kmp算法实践题

    给定两个由英文字母组成的字符串 String 和 Pattern,要求找到 Pattern 在 String 中第一次出现的位置,并将此位置后的 String 的子串输出.如果找不到,则输出“Not ...

随机推荐

  1. python3函数进阶

    1.命名空间和作用域 命名空间     加载         内置命名空间             python解释器自带的变量和函数             开启python解释器自动加载内置命名空 ...

  2. 6.反编译 java---class (字节码文件)---反编译(IDEA):

  3. 【Java面试】Mybatis篇

    1.MyBatis编程步骤是什么样的? ① 创建SqlSessionFactory ② 通过SqlSessionFactory创建SqlSession ③ 通过sqlsession执行数据库操作 ④  ...

  4. tensorboard网络结构

    一.tensorboard网络结构 import tensorflow as tffrom tensorflow.examples.tutorials.mnist import input_data ...

  5. python+win32--com实现excel自动化

    import win32com  APP_TYPE = 'Excel.Application'  xlBlack,xlRed,xlGray,xlBlue = 1,3,15,41 xlBreakFull ...

  6. for in 和 for i 十月 javascript 第一弹 记录

    for in 里面的 i 不仅有 数字还有这些鬼

  7. 龙芯 Fedora 28 设置 VNC

    系统为龙芯版Fedora28 (床28) Fedora防火墙默认阻止了VNC所需的端口的访问,导致VNC客户端一直无法连接. 安装VNC Server sudo dnf install tigervn ...

  8. Golang - 指针与引用

    ​ Golang有指针 , 那么一切数据都是值传递吗 ? 都需要用户进行指针传递吗, 其实不然, 对于Go语言, 虽然有指针, 但是其也有引用传递. 是不是很绕, 因为引用传递就是指针传递哇 . 我们 ...

  9. Shell 脚本学习总结

    自己根据目前学到的东西制作了一张思维导图

  10. jdk for centos7

    https://www.cnblogs.com/chy123/p/6750351.html