题意

给你一个数字序列S,再给一个数字序列pattern,S和pattern中的数字都是1到s(s<=25)。每个序列里的数字都有个排名,也就是第几小,现在我们要用pattern来匹配S。在本题中匹配是指每个数字的排名都一样,即同是第k小,最后输出匹配的所有位置。

思路

KMP好题,对KMP的理解又透彻了一点点~

我们考虑两个字符串A,B

在此题中,A[1],A[2],…,A[k]与B[1],B[2],…,B[k]匹配条件:

若A[1],A[2],…,A[k-1]与B[1],B[2],…,B[k-1]匹配,则加上A[k]与B[k]仍然匹配的条件是:

必须在与k无关的时间复杂度内完成该操作

然而,由于A[1],A[2],…,A[k-1]与B[1],B[2],…,B[k-1]已经匹配,可以简化比较操作
考虑到字符集很小(不超过25),我们可以定义以下几个变量:

Occur[p],Low[i],High[i]可能不存在,若存在则取最小的符合条件的x

则A[1],A[2],…,A[k-1]与B[1],B[2],…,B[k-1]匹配,加上A[k]与B[k]仍然匹配的条件可简化为:

接下来就可以用O(SK)时间内预处理求出Occur[p],Low[i],High[i],然后套用kmp算法,总时间复杂度为O(N+SK)

代码

[cpp]
#include <iostream>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <string>
#include <cstring>
#include <vector>
#include <set>
#include <stack>
#include <queue>
#define MID(x,y) ((x+y)/2)
#define MEM(a,b) memset(a,b,sizeof(a))
#define REP(i, begin, end) for (int i = begin; i <= end; i ++)
using namespace std;

typedef long long LL;
int n, k, s;
int S[100005], pattern[25005];
int low[25005], high[25005], ranks_pos[30], next[100005];
vector <int> ans;

void init(){
MEM(ranks_pos, -1); MEM(low, -1); MEM(high, -1);
for (int i = 0; i < k; ++ i){
int j = pattern[i] - 1;
while(j){
if (~ranks_pos[j]){ low[i] = j; break; }
j --;
}
j = pattern[i] + 1;
while(j < s+1){
if (~ranks_pos[j]){ high[i] = j; break; }
j ++;
}
if (ranks_pos[pattern[i]] == -1)
ranks_pos[pattern[i]] = i;
}
}
bool check(int A[], int B[], int k, int pos){
if (~ranks_pos[A[pos]]){
if (B[k+ranks_pos[A[pos]]] != B[k+pos]) return false;
}
if (~low[pos] && ~ranks_pos[low[pos]]){
if (B[k+ranks_pos[low[pos]]] >= B[k+pos]) return false;
}
if (~high[pos] && ~ranks_pos[high[pos]]){
if (B[k+ranks_pos[high[pos]]] <= B[k+pos]) return false;
}
return true;
}
void get_next(){
int j = -1;
next[0] = -1;
for (int i = 1; i < k; ++ i){
while(j > -1 && !check(pattern, pattern, i-j-1, j+1)) j = next[j];
if (check(pattern, pattern, i-j-1, j+1)) j ++;
next[i] = j;
//printf("next%d = %d\n", i, next[i]);
}
}
void kmp(){
ans.clear();
get_next();
int j = -1;
for (int i = 0; i < n; ++ i){
while(j > -1 && !check(pattern, S, i-j-1, j+1)) j = next[j];
if (check(pattern, S, i-j-1, j+1)) j ++;
if (j == k - 1){
ans.push_back(i - k + 1);
j = next[j];
}
}
}
int main(){
while(scanf("%d %d %d", &n, &k, &s) != EOF){
for (int i = 0; i < n; i ++) scanf("%d", &S[i]);
for (int i = 0; i < k; i ++) scanf("%d", &pattern[i]);
init();
kmp();
printf("%d\n", ans.size());
for (int i = 0; i < (int)ans.size(); ++ i){
printf("%d\n", ans[i]+1);
}
}
return 0;
}
[/cpp]

POJ 3167 Cow Pattern ★(KMP好题)的更多相关文章

  1. POJ 3167 Cow Patterns (KMP+前缀和)

    题意:给你两串数字,长度分别为n和m,数字大小在[1,25].当后一串数字每个数字的排名位置与前一串数字(任一长度为m的子串)每个数字的排名位置一致时就完全匹配,最后求哪些位置是完全匹配的. 例如:1 ...

  2. POJ 3167 Cow Patterns(模式串浮动匹配)

    题目链接:http://poj.org/problem?id=3167 题意:模式串可以浮动的模式匹配问题给出模式串的相对大小,需要找出模式串匹配次数和位置. 思路:统计比当前数小,和于当前数相等的, ...

  3. POJ:3461-Oulipo(KMP模板题)

    原题传送:http://poj.org/problem?id=3461 Oulipo Time Limit: 1000MS Memory Limit: 65536K Description The F ...

  4. POJ 3176 Cow Bowling (水题DP)

    题意:给定一个金字塔,第 i 行有 i 个数,从最上面走下来,只能相邻的层数,问你最大的和. 析:真是水题,学过DP的都会,就不说了. 代码如下: #include <cstdio> #i ...

  5. poj 3461 Oulipo(KMP模板题)

    Oulipo Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 36903   Accepted: 14898 Descript ...

  6. POJ 3461 Oulipo(KMP裸题)

    Description The French author Georges Perec (1936–1982) once wrote a book, La disparition, without t ...

  7. POJ Oulipo KMP 模板题

    http://poj.org/problem?id=3461 Oulipo Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4 ...

  8. POJ Oulipo(KMP模板题)

    题意:找出模板在文本串中出现的次数 思路:KMP模板题 #include<cstdio> #include<cstring> #include<cmath> #in ...

  9. POJ 3045 Cow Acrobats (贪心)

    POJ 3045 Cow Acrobats 这是个贪心的题目,和网上的很多题解略有不同,我的贪心是从最下层开始,每次找到能使该层的牛的风险最小的方案, 记录风险值,上移一层,继续贪心. 最后从遍历每一 ...

随机推荐

  1. python3 爬虫之Pyquery的使用方法

    安装 pip install pyquery 官方文档: https://pythonhosted.org/pyquery/ 初始化方式(四种) 1. 直接字符串 from pyquery impor ...

  2. delphi pchar 指针错误

    2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 var P: Pchar; //P 是指针 CSize: Cardinal; ...

  3. 03 linux命令的操作

    开启Linux操作系统,要求以root用户登录GNOME图形界面,语言支持选择为汉语 使用快捷键切换到虚拟终端2,使用普通用户身份登录,查看系统提示符 使用快捷键切换到虚拟终端5,使用管理员身份登录, ...

  4. HDU 1142 A Walk Through the Forest(Dijkstra+记忆化搜索)

    题意:看样子很多人都把这题目看错了,以为是求最短路的条数.真正的意思是:假设 A和B 是相连的,当前在 A 处, 如果 A 到终点的最短距离大于 B 到终点的最短距离,则可以从 A 通往 B 处,问满 ...

  5. mysql 及练习题

    if() 函数的用法 IF(expr1,expr2,expr3),如果expr1的值为true,则返回expr2的值,如果expr1的值为false, mysql,'女','男') as sex fr ...

  6. Java并发编程:Lock(转)

    本文转自:http://www.cnblogs.com/dolphin0520/p/3923167.html Java并发编程:Lock 在上一篇文章中我们讲到了如何使用关键字synchronized ...

  7. 本地多张图片采用jmeter上传到ftp服务器的方法和获取服务器日志中某些关键字的基本方法

    测试需求: 本地图片上传到ftp服务器里和另外两台不同算法比对服务器进行比对,得出漏检和误检结果:这实际属于功能测试范畴. 测试思路: 第一种方法:使用实际场景的摄像机抓拍图片上传到服务器,用录屏软件 ...

  8. RabbitMQ学习之(五)_一个基于PHP的RabbitMQ操作类

    //amqp.php类文件 <?php class Amqp { public $e_name; public $q_name; public $k_route; public $channel ...

  9. CentOS系统下yum命令的详细使用方法

    yum是什么yum = Yellow dog Updater, Modified 主要功能是更方便的添加/删除/更新RPM包. 它能自动解决包的倚赖性问题. 它能便于管理大量系统的更新问题 yum特点 ...

  10. The mind

    Youtube励志红人Mateusz M,最近刚推出了网友期待已久最新励志短片<The mind>,短短一周播放量已近50万.来自波兰的他,年仅23岁,用自己擅长的蒙太奇,已创作十几个影响 ...