题目
给定 n 个长度不超过 1010 的字符串以及 m 次询问,每次询问给出一个字符串和一个操作次数上限。
对于每次询问,请你求出给定的 n 个字符串中有多少个字符串可以在上限操作次数内经过操作变成询问给出的字符串。
每个对字符串进行的单个字符的插入、删除或替换算作一次操作。
输入格式
第一行包含两个整数 n 和 m。
接下来 n 行,每行包含一个字符串,表示给定的字符串。
再接下来 m 行,每行包含一个字符串和一个整数,表示一次询问。
字符串中只包含小写字母,且长度均不超过 1010。
输出格式
输出共 m 行,每行输出一个整数作为结果,表示一次询问中满足条件的字符串个数。
数据范围
1≤n,m≤1000,
输入样例:
3 2
abc
acd
bcd
ab 1
acbd 2
输出样例:
1
3
AC代码
//最短编辑距离应用题
#include<bits/stdc++.h>
using namespace std;
const int N = 1010, M = 13;
int f[M][M];
char s[N][M];
int n, m;
int mindo(char a[], char b[]){ //最短编辑距离DP代码
int alen = strlen(a + 1), blen = strlen(b + 1);
for(int i = 1; i <= alen; i++)f[i][0] = i;
for(int j = 1; j <= blen; j++)f[0][j] = j;
for(int i = 1; i <= alen; i++)
for(int j = 1; j <= blen; j++)
{
f[i][j] = min(f[i - 1][j], f[i][j- 1]) + 1;
f[i][j] = min(f[i][j], f[i - 1][j - 1] + (a[i] != b[j]));
}
return f[alen][blen];
}
int main(){
cin >> n >> m;
for(int i = 1; i <= n; i++)scanf("%s", s[i] + 1);
while(m--)
{
int lim = 0;
char str[N];
scanf("%s%d", str + 1, &lim);
int res = 0;
for(int i = 1; i <= n; i++) //遍历已有数组
if(mindo(s[i], str) <= lim)res++; //可以在限制次数内完成,计数
cout << res << endl;
}
return 0;
}