Message CodeForces - 157C 滑动门技巧

传送门

题目大意: 有一个字符串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;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值