2016西电校赛网络赛 Problem C 寻找万神

Problem C

问题

某人在教育学弟时说:“电院的万神,比你们高的不知道哪里去了,我和他
谈笑风生!”但是学弟 too young,too simple,根本不认识万神,只好自己到百
度上搜。
为了衡量搜索结果和万神的相关程度,学弟希望知道一篇文章中“万神”二
字出现的次数。你能帮助他吗?

输入格式

输入包含多组数据,请处理到文件结束。
每组数据,第一行包含整数 n,表示这组数据的行数。之后 n 行表示学弟搜
到的文章。为了避免中文编码问题,文章用拼音给出。
保证文章只包含大写英文字母(英文)、小写英文字母(拼音)、句号 “.” 、
逗号”,” 。
保证 1 ≤ n ≤ 100,输入文件总长度至多是 5MB 。

输出格式

对于每组数据,输出文章中“万神”二字(拼音为 “wanshen”,不含引号)
的出现次数。

输入样例

5
wanshenshixidianACM
dediyidashen.erqiew
anshendeDOTAyedadeh
enhao.womendouhenOR
Zwanshen.
1
WANSHEN.

输出样例

3
0

样例解释

对于第一组样例,注意“万神”跨越了文章的第 2 行和第 3 行。
对于第二组样例,我们不认为英文字母序列 “WANSHEN” 表示万神。

思路

KMP模板,,最长公共前后缀

/************************************************************************* > File Name: c.cpp > Author: dulun > Mail: [email protected] > Created Time: 2016年04月20日 星期三 13时04分24秒 ************************************************************************/

#include<iostream>
#include<stdio.h>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#define LL long long
using namespace std;

const int N = 50086;
int nxt[10];
char a[101][N];
char p[10] = "wanshen";
int n;
int l = strlen(p);
void init()
{
    int k = 0;
    nxt[0] = 0;
    for(int i = 1; i < l; i++)
    {
        while(k && p[i] != p[k]) k = nxt[k-1];
        if(p[i] == p[k]) k++;
        nxt[i] = k;
    }
}

int kmp()
{
    int ans = 0;
    int k = 0;
    for(int i = 0; i < n; i++)
    {
        int len = strlen(a[i]);
        for(int j = 0; j < len; j++)
        {
            while(k && a[i][j] != p[k]) k = nxt[k-1];
            if(p[k] == a[i][j]) k++;
            if(k == l)
            {
                ans++;
                k = 0;
            }
        }
    }
    return ans;
}

int main()
{
    init();
    while(cin>>n)
    {
        for(int i = 0; i < n; i++)
        {
            scanf("%s", a[i]);
        }
        printf("%d\n", kmp());
    }
    return 0;
}

你可能感兴趣的:(C语言,KMP)