POJ 3461__KMP算法
【题目描述】
法国作家乔治·佩雷克(Georges Perec,1936-1982)曾经写过一本书,《敏感字母》(La disparition),全篇没有一个字母‘e’。他是乌力波小组(Oulipo Group)的一员。下面是他书中的一段话:
Tout avait Pair normal, mais tout s’affirmait faux. Tout avait Fair normal, d’abord, puis surgissait l’inhumain, l’affolant. Il aurait voulu savoir où s’articulait l’association qui l’unissait au roman : stir son tapis, assaillant à tout instant son imagination, l’intuition d’un tabou, la vision d’un mal obscur, d’un quoi vacant, d’un non-dit : la vision, l’avision d’un oubli commandant tout, où s’abolissait la raison : tout avait l’air normal mais…
佩雷克很可能在下面的比赛中得到高分(当然,也有可能是低分)。在这个比赛中,人们被要求针对一个主题写出甚至是意味深长的文章,并且让一个给定的“单词”出现次数尽量少。我们的任务是给评委会编写一个程序来数单词出现了几次,用以得出参赛者最终的排名。参赛者经常会写一长串废话,例如500000个连续的‘T’。并且他们不用空格。
因此我们想要尽快找到一个单词出现的频数,即一个给定的字符串在文章中出现了几次。更加正式地,给出字母表{'A','B','C',...,'Z'}和两个仅有字母表中字母组成的有限字符串:单词W和文章T,找到W在T中出现的次数。这里“出现”意味着W中所有的连续字符都必须对应T中的连续字符。T中出现的两个W可能会部分重叠。
【输入格式】
输入包含多组数据。
输入文件的第一行有一个整数,代表数据组数。接下来是这些数据,以如下格式给出:
第一行是单词W,一个由{'A','B','C',...,'Z'}中字母组成的字符串,保证1<=|W|<=10000(|W|代表字符串W的长度)
第二行是文章T,一个由{'A','B','C',...,'Z'}中字母组成的字符串,保证|W|<=|T|<=1000000。
【输出格式】
对每组数据输出一行一个整数,即W在T中出现的次数。
【样例输入】
3
BAPC
BAPC
AZA
AZAZAZA
VERDI
AVERDXIVYERDIAN
【样例输出】
1
3
0
——————————————————————————————————————————————————————————————
题意:查找子串在主串中出现的次数。
子串查找匹配:KMP.
KMP算法初看非常晦涩难懂,网上代码的思想都是一样的,但是写法上有一些细节的不同,也就使得它更难掌握。在这里我会介绍一些我的理解,不一定好,只是自己的学习记录。如果不懂可以看下面的博客:http://blog.csdn.net/u011564456/article/details/20862555?utm_source=tuicool&utm_medium=referral ,非常通俗易懂。
KMP算法的思想:
普通的子串匹配常用的方法为:子串与主串比较,如果过程中失败则后移一位。如:
主串:a b c a d a b c a b
子串:a b c a b
子串: a b c a b (后移后位置)
这样时间复杂度为(n*m);
三位大神分析发现,我们不必一位一位的移动,因为后移一位勘定无法匹配。而是应该把子串直接移到3号位(从0开始),从主串4号位与子串1号位开始比较,这样就加快了速度。而移动到几号位和子串中比较失败前的前缀中的公共前后缀长度有关,如比较到子串中的4号位b时失败,则前缀为 a b c a ,这样它的公共前后缀为a,所以从a后面的字符开始比较就可以了,a不用比了。主串中前面的位置也不用比了。这样任务就成了如何快速的求公共前后缀的长度,即next[]。
next[]的含义:
前面说到next[]为公共前后缀的长度,但是有一点点不同,应该这样理解:公共前后缀的前缀最后一个字符的序号。
如:next[i]表示子串前i+1个字符组成的前缀(0——i)的公共前后缀的前缀的最后一个字符的序号。如a b c a b 的next[]=-1、-1、-1、0、1。
next[]的求法:
next[0]=-1;
求f[i]时,如果zichuan[i]==zichuan[next[i-1]+1],即子串0——i-1组成的前缀的公共前后缀的前缀后的一个字符与后缀后的一个字符相同,则next[后缀后的一个字符的序号]=next[后缀的最后一个字符的序号]+1。

如上图子串中:已知next[i-1]为第一块绿色部分,而第二块绿色部分与之相同。如果第一块红色部分与第二块红色部分也相同则next[i]=next[i-1]+1;
如果zichuan[i]!=zichuan[next[i-1]+1],则next[i]肯定比next[i-1]小,为从“zichuan[0]开始的”一个与“从zichuan[i]往前“等长的串。而由于next[i-1]的存在,”从zichuan[i-1]往前“的串与”zichuan[next[i]]往前“等长的串相同。所以next[i]=next[next[i-1]]+1;当然,还有可能递归下去。

如上图子串中,next[i-1]为第一块绿色部分,第二块绿色部分与之相同。但是两块红色部分不相同,即zichuan[next[i-1]+1]!=zichuan[i]。所以next[i]的后缀只能是第二块红色部分和前面绿色块的一部分,假设是它前面加黑的的部分。由于next[i-1]的存在,第一块红色部分前面肯定也有有一块加黑的的部分。因为next[i]的存在,子串的开始部分肯定也有一段加黑点的部分,而且一个与zichuan[i]相同的字符(加红点的)(如果没有相同的字符,则需要继续递归!)。因此可以看出前两个加黑点的部分是相同的,zichuan[i]与加红点字符是相同的。也就是说next[i]==next[next[i-1]]+1。
样例代码:
void getnext()//next[]数组
{
next[0]=-1;
for(int i=1,j;i<l;++i)
{
j=next[i-1];
while(s[i]!=s[j+1] && j>=0)j=next[j];
next[i]=s[i]==s[j+1]?next[j]+1:-1;
}
}
int KMPf()//子串出现的位置
{
int i=0,j=0;
while(i<l&&j<ll)
{
if(s[i]==ss[j])
{
i++;j++;
}
else
if(i==0)j++;
else i=next[i-1]+1;
}
return i==l?j-l:-1;
}
int KMP()//统计子串的次数
{
int i=0,j=0;
while(i<l&&j<ll)
{
if(s[i]==ss[j])
{
i++;j++;
}
else
if(i==0)j++;
else i=next[i-1]+1;
if(i==l)
{
ans++;i=next[i-1]+1;
}
}
return ans;
}
——————————————————————————————————————————————————————————————
1 #include<cstdio>
2 #include<cstring>
3 #include<iostream>
4 #include<algorithm>
5
6 using namespace std;
7 char s[10005],ss[1000005];
8 int l,ll,n,ans;
9 int next[10005];
10 void init()
11 {
12 scanf("%s%s",s,ss);
13 l=strlen(s);ll=strlen(ss);
14 ans=0;
15 }
16 void getnext()
17 {
18 next[0]=-1;
19 for(int i=1,j;i<l;++i)
20 {
21 j=next[i-1];
22 while(s[i]!=s[j+1] && j>=0)j=next[j];
23 next[i]=s[i]==s[j+1]?j+1:-1;
24 }
25 }
26 int KMP()
27 {
28 int i=0,j=0;
29 while(i<l&&j<ll)
30 {
31 if(s[i]==ss[j])
32 {
33 i++;j++;
34 }
35 else
36 if(i==0)j++;
37 else i=next[i-1]+1;
38 if(i==l)
39 {
40 ans++;i=next[i-1]+1;
41 }
42 }
43 return ans;
44 }
45 int main()
46 {
47 freopen("oulipo.in","r",stdin);
48 freopen("oulipo.out","w",stdout);
49 scanf("%d",&n);
50 while(n--)
51 {
52 init();
53 getnext();
54 printf("%d\n",KMP());
55 }
56 fclose(stdin);
57 fclose(stdout);
58 return 0;
59 }
POJ 3461__KMP算法的更多相关文章
- poj 1523Tarjan算法的含义——求取割点可以分出的连通分量的个数
poj 1523Tarjan算法的含义——求取割点可以分出的连通分量的个数 题目大意:如题目所示 给你一些关系图——连通图,想要问你有没有个节点,损坏后,可以生成几个互相独立的网络(也就是连通分量), ...
- POJ题目算法分类总结博客地址
http://blog.csdn.net/sunbaigui/article/details/4421705 又从这个地址找了一些:http://blog.csdn.net/koudaidai/art ...
- STL(pair map set vector priority_queue) poj 3297
POJ 3297 算法竞赛初级杂烩包 题意:学生选课,没个学生只能选一门课.大写字符是课的名字,小写是人名.如果课程后面有多个相同名字算一个,如果一个人选多门课,则他选不上课,输出课和每门课选课人数 ...
- 浅谈KMP算法及其next[]数组
KMP算法是众多优秀的模式串匹配算法中较早诞生的一个,也是相对最为人所知的一个. 算法实现简单,运行效率高,时间复杂度为O(n+m)(n和m分别为目标串和模式串的长度) 当字符串长度和字符集大小的比值 ...
- [ACM训练] 算法初级 之 搜索算法 之 深度优先算法DFS (POJ 2251+2488+3083+3009+1321)
对于深度优先算法,第一个直观的想法是只要是要求输出最短情况的详细步骤的题目基本上都要使用深度优先来解决.比较常见的题目类型比如寻路等,可以结合相关的经典算法进行分析. 常用步骤: 第一道题目:Dung ...
- [ACM训练] 算法初级 之 搜索算法 之 广度优先算法BFS (POJ 3278+1426+3126+3087+3414)
BFS算法与树的层次遍历很像,具有明显的层次性,一般都是使用队列来实现的!!! 常用步骤: 1.设置访问标记int visited[N],要覆盖所有的可能访问数据个数,这里设置成int而不是bool, ...
- [ACM训练] 算法初级 之 基本算法 之 枚举(POJ 1753+2965)
先列出题目: 1.POJ 1753 POJ 1753 Flip Game:http://poj.org/problem?id=1753 Sample Input bwwb bbwb bwwb bww ...
- 算法手记 之 数据结构(线段树详解)(POJ 3468)
依然延续第一篇读书笔记,这一篇是基于<ACM/ICPC 算法训练教程>上关于线段树的讲解的总结和修改(这本书在线段树这里Error非常多),但是总体来说这本书关于具体算法的讲解和案例都是不 ...
- 算法手记 之 数据结构(堆)(POJ 2051)
一篇读书笔记 书籍简评:<ACM/ICPC 算法训练教程>这本书是余立功主编的,代码来自南京理工大学ACM集训队代码库,所以小编看过之后发现确实很实用,适合集训的时候刷题啊~~,当时是听了 ...
随机推荐
- [leetcode]720. Longest Word in Dictionary字典中最长的单词
b.compareTo(a) 这个函数是比较两个值得大小,如果b比a大,那么返回1 如果小,那么返回-1,相等返回0 如果比较的是字符串,那么比较字典编纂顺序,b靠前返回-1,靠后返回1 这个题的核心 ...
- 在开发板上显示英文字符和汉字--tiny6410
程序字符需要改成gb2312.否则无法正常显示中文字符. main.c代码: #include <sys/types.h> #include <sys/stat.h> #inc ...
- package和import机制
package是Java中的包机制,包机制的作用是方便为了程序的管理.不同功能的类机制分别存放在不同的包下面.(按照功能划分,不同的包有着不同的性质) package怎么使用:package是一个关键 ...
- linux学习之--虚拟机安装linux【centerOS】
计划把学习中的软件安装使用记录下来,以下是使用VMware 按照 Linux 使用桥接网络虚拟机和windows中都有不同的ip地址
- python-scrapy爬取某招聘网站(二)
首先要准备python3+scrapy+pycharm 一.首先让我们了解一下网站 拉勾网https://www.lagou.com/ 和Boss直聘类似的网址设计方式,与智联招聘不同,它采用普通的页 ...
- 使用sqoop将mysql数据导入到hive中
首先准备工具环境:hadoop2.7+mysql5.7+sqoop1.4+hive3.1 准备一张数据库表: 接下来就可以操作了... 一.将MySQL数据导入到hdfs 首先我测试将zhaopin表 ...
- 基于http的netty demo
1.引入netty的pom <dependency> <groupId>io.netty</groupId> <artifactId>netty-all ...
- Java ClassLoader浅析
双亲委派 提起 java 类加载器,自然绕不开其双亲委派模型 什么是双亲委派 提起双亲委派,首先想到便是那张经典的向上委派图 一般场景下,当某个类将要被加载时,由系统上下文默认的类加载器Thread. ...
- Azure Terraform(三)部署 Web 应用程序
一,引言 上一节关于 Terraform 的文章讲到 Terraform 使用到的一些语法,以及通过演示使用 Terraform 在Azure 上部署资源组,极大的方便了基础设施实施人员,也提高了基础 ...
- 7.shell脚本编程
1.shell 脚本语言的基本用法 1.1shell 脚本创建 1.格式要求:首行shebang机制 #!/bin/bash #!/usr/bin/python #!/usr/bin/perl 2.添 ...