传送门
题目大意: 有一个字符串s, 从中任取一个子串, 然后改变尽量少的次数使它变成字符串u。
改变规则如下(修改取的子串, 不是字符串s):
在字符串首或尾添加一个字符
在字符串任意一个位置修改一个字符
求修改的最少次数
解题思路: 找一个子串,与目标串对应位置最多有几个相同的(也就是要改动最少的), 这样我们总是匹配与u的长度一样长的字符串即可, 但是有个问题就是, 如果u串中后半部分与s串的开头匹配就被忽略了, 因此我们假想s前面还有一个长度为u长度的空串, 让u从这个空串的开头开始匹配, 只要是空串那么一定需要补一个字符。
代码:
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
char s[2050], u[2050];
int main()
{
scanf("%s%s", s, u);
int ls = strlen(s);
int lu = strlen(u);
int ans = lu;
int j, cur;
for(int i=0; i<ls+lu; ++i)
{
for(cur=j=0; j<lu; ++j) //根据u的长度挨个匹配看看是否相同, 如果不相同就要改动一次
{
//i 为从空串开头开始, i-lu就是s的下标,i-lu+j < 0 表示当前还在空串, 因此需要补一个字符。
//如果i-lu +j >= ls 表示长度超越了s的长度, 因此需要在后面补一个字符
//在s的范围之内,就看是否相等了, 如果不想等就要改变一个字符
cur += (i-lu+j >= 0 && i+j < lu+ls) ? s[i-lu+j] != u[j] : 1;
}
ans = min(ans, cur);
}
cout << ans << endl;
return 0;
}