kmp 剪花布条 HDU - 2087

地址 https://vjudge.ppsucxtt.cn/problem/HDU-2087


一块花布条,里面有些图案,另有一块直接可用的小饰条,里面也有一些图案。对于给定的花布条和小饰条,计算一下能从花布条中尽可能剪出几块小饰条来呢?

Input
输入中含有一些数据,分别是成对出现的花布条和小饰条,其布条都是用可见ASCII字符表示的,可见的ASCII字符有多少个,
布条的花纹也有多少种花样。花纹条和小饰条不会超过1000个字符长。如果遇见#字符,则不再进行工作。
Output
输出能从花纹布中剪出的最多小饰条个数,如果一块都没有,那就老老实实输出0,每个结果之间应换行。
Sample Input
abcde a3
aaaaaa aa
#
Sample Output
0
3

解法

其实使用暴力查找也是可以过的


#include <iostream>
#include <vector>
#include <string>

using namespace std;

/*
abcde a3
aaaaaa aa
#
*/

int main()
{
while(1){
  string a, b;
  cin >> a;
  if (a == "#") return 0;
  cin >> b;
  size_t pos = 0; int count = 0;
  while (pos < a.size()) {
    pos = a.find(b, pos);
    if (pos == string::npos) { break; }
      count++;
      pos += b.size();
    }
    cout << count << endl;
  }
  return 0;
}

使用kmp模板解答,唯一需要注意的是,kmp是尽量匹配每个字符串,能够重叠,但是本题的剪花布条是进行匹配 但是不能重叠

需要做一点小改动


#include <iostream>
#include <vector>
#include <string>

using namespace std;

/*
abcde a3
aaaaaa aa
#
*/

int ne[2000];
char a[2000];
char b[2000];

int main()
{
while(1){
cin >> (a + 1);
if (a[1] == '#') break;
cin >> (b+1);
int m = strlen(a + 1);
int n = strlen(b + 1);

for (int i = 2, j = 0; i <= n; i++) {
  while (j && b[i] != b[j + 1]) j = ne[j];
  if (b[i] == b[j + 1]) j++;
  ne[i] = j;
}
int count = 0;
for (int i = 1, j = 0; i <= m; i++) {
    while (j && a[i] != b[j + 1]) j = ne[j];
      if (a[i] == b[j + 1]) j++;
      if (j == n) {
        count++;
        //j = ne[j];
        j = 0;
      }
    }
    cout << count << endl;
  }
  return 0;
}

我的视频题解空间

posted on 2021-10-04 11:32  itdef  阅读(55)  评论(0编辑  收藏  举报

导航