CF1096D Easy Problem 题解

题目链接

我们设 f [ i ] [ j ] f[i][j] f[i][j] 为让原字符串满足最多包含 hard \texttt{hard} hard 的前 j j j 个字符,所需要的最小修改代价
特别的, f [ i ] [ 0 ] f[i][0] f[i][0] 表示不包含 h h h 的最小修改代价

考虑转移方程,有两种情况

s [ i ] ≠ c [ j + 1 ] s[i] \ne c[j+1] s[i]=c[j+1] 时( c c c 是预先处理好的 hard \texttt{hard} hard),加上第 i i i 个字符后不会增加最大匹配的长度,所以 f [ i ] [ j ] = f [ i − 1 ] [ j ] f[i][j]=f[i-1][j] f[i][j]=f[i1][j]

s [ i ] = c [ j + 1 ] s[i] = c[j+1] s[i]=c[j+1],又可以选择删与不删第 i i i 个字符
所以 f [ i ] [ j ] = min ⁡ ( f [ i − 1 ] [ j ] + a [ i ] , f [ i − 1 ] [ j − 1 ] ) f[i][j]=\min(f[i-1][j]+a[i],f[i-1][j-1]) f[i][j]=min(f[i1][j]+a[i],f[i1][j1])
在选择不删除的时候,为了不凑出 hard \texttt{hard} hard 的前 j + 1 j+1 j+1 个字符,我们在前面最多只能凑出前 j − 1 j-1 j1(使得 c [ j + 1 ] c[j+1] c[j+1] 与前面的字符断开)。

我们发现在每次转移的时候,只用到了 f [ i − 1 ] f[i-1] f[i1] f [ i ] f[i] f[i] 这两层,所以我们可以把第一维用滚动数组省略掉

#include
#include
using namespace std;
const long long Maxn=100000+10,inf=(1ll<<60);
long long f[5],a[Maxn];
char c[5]={'f','h','a','r','d'};
char s[Maxn];
long long n,ans=inf;
int main()
{
//	freopen("in.txt","r",stdin);
	scanf("%lld",&n);
	scanf("%s",s+1);
	for(long long i=1;i<=n;++i)
	scanf("%lld",a+i);
	for(long long i=1;i<=n;++i)
	for(long long j=3;j>=0;--j)
	if(s[i]==c[j+1])
	{
		f[j]+=a[i];
		if(j)f[j]=min(f[j],f[j-1]);
	}
	for(long long i=0;i<4;++i)
	ans=min(ans,f[i]);
	printf("%lld\n",ans);
	return 0;
}

你可能感兴趣的:(题解,#Codeforces)