String Painter, Chengdu 2008, LA4394
题目链接:https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=2395
题目大意:给定两个长度相等,只有小写字母组成的字符串s和t,每步可以把s的一个连续字串“刷“成同一个字母,问至少需要多少步才能把s变成t。比如:s = bbbbbbb, t = aaabccb,最少需要两步可实现将s变成t:bbbbbbb --> aaabbbb --> aaabccb。(字符串长度不超过100)
题解:由于题目历史较为久远,网上鲜有关于此题的题解。以下内容仅个人YY之作,难登大雅之堂。
初见此题有DP的即视感,由于”刷“这个操作随意性强,没有左右的顺序,因此排除一般的线性DP,考虑区间DP。
我认为DP难度在于选择状态,而状态的选取通常与题目的特殊性质紧密相连。而构造状态的技巧在于,考虑怎样的大状态能由小状态更新过来,更新方法与题目性质挂钩。
首先预处理出p[l][r][ch]对于s1(终状态)区间[l,r]上底色为ch的情况下,最少需要几步操作变为s1上的对应状态。
p[l][r][ch] = p[l + 1][r - 1][ch](s1[l] == ch)
p[l][r][ch] = min{p[l][k][s1[l]] + p[k + 1][j][ch] + 1} (s1[l] != ch)
f[l][r]表示从l到r这段区间,由s2变为s1需要多少步操作。与题目中”刷”的操作挂钩,考虑从l开始刷k个格子全部变为s1[l],这样[l,k]上已经涂满了底色s1[l],后续操作数由p[l][k][s1[l]]转移。
f[l][r] = f[l + 1][r](s1[l] == s2[l])
f[l][r] = min{p[l][r][s1[l]] + f[k + 1][r] + 1} (s1[l] != s2[l])
时间复杂度O(len*len*len*26),以为会T,没想到跑得还挺快的…
//By Ro_mantic 2015-11-04
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <ctime>
#include <cmath>
#include <cctype>
#include <climits>
#include <string>
#include <algorithm>
using namespace std; const int MaxN = ;
char s1[MaxN + ], s2[MaxN + ];
int len, len1, len2, Minn, p[MaxN + ][MaxN + ][MaxN],
f[MaxN + ][MaxN + ]; void Prepare()
{
len1 = strlen(s1);
for (int i = ; i <= len1 - ; i++)
for (int j = 'a'; j <= 'z'; j++)
if (s1[i] == j) p[i][i][j] = ;
else p[i][i][j] = ;
for (int len = ; len <= len1; len++)
for (int l = ; l <= len1 - len; l++)
for (int ch = 'a'; ch <= 'z'; ch++)
{
int r = l + len - ;
if (s1[l] == ch) p[l][r][ch] = p[l + ][r][ch];
else
{
Minn = p[l + ][r][ch] + ;
for (int k = l + ; k <= r; k++)
Minn = min(p[l + ][k][s1[l]] + p[k + ][r][ch] + , Minn);
p[l][r][ch] = Minn;
}
}
//printf("%d\n", p[0][8]['a']);
} void Solve()
{
len1 = strlen(s1);
for (int i = ; i <= len1 - ; i++)
if (s1[i] == s2[i]) f[i][i] = ; else f[i][i] = ;
for (int len = ; len <= len1; len++)
for (int l = ; l <= len1 - len; l++)
{
int r = l + len - ;
if (s1[l] == s2[l]) f[l][r] = f[l + ][r];
else
{
Minn = f[l + ][r] + ;
for (int k = l + ; k <= r; k++)
Minn = min(p[l + ][k][s1[l]] + f[k + ][r] + , Minn);
f[l][r] = Minn;
}
}
printf("%d\n", f[][len1 - ]);
} int main()
{
while (~scanf("%s", s2))
{
scanf("%s", s1);
Prepare();
Solve();
}
}
String Painter, Chengdu 2008, LA4394的更多相关文章
- HDOJ 题目2474 String painter(区间DP)
String painter Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- HDU2476 String painter
题意 String painter Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- HDU2476 String painter —— 区间DP
题目链接:https://vjudge.net/problem/HDU-2476 String painter Time Limit: 5000/2000 MS (Java/Others) Me ...
- hdu 2476 (string painter) ( 字符串刷子 区间DP)
String painter Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- hdu2476 String painter(区间dp)
题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=2476 Problem Description There are two strings ...
- uva live 4394 String painter 间隔dp
// uva live 4394 String painter // // 问题是,在培训指导dp运动主题,乍一看,我以为只是一点点复杂 // A A磕磕磕,两个半小时后,.发现超过例子.然而,鉴于他 ...
- HDU 2476 String painter(区间DP)
String painter Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...
- 刷题总结——String painter(hdu2476)
题目: Problem Description There are two strings A and B with equal length. Both strings are made up of ...
- String painter(区间DP)
There are two strings A and B with equal length. Both strings are made up of lower case letters. Now ...
随机推荐
- 主席树的各类模板(区间第k大数【动,静】,区间不同数的个数,区间<=k的个数)
取板粗 好东西来的 1.(HDOJ2665)http://acm.hdu.edu.cn/showproblem.php?pid=2665 (POJ2104)http://poj.org/probl ...
- Java的观察者
class Teacher extends Observable { public void startLesson() { System.out.println(String.format(&quo ...
- Linux系统lvm管理
pv: 物理卷,被pv命令处理过的物理分区vg:物理卷组 被组装到一起的物理卷pe: 物理扩展 lvm设备的最小存储单元 lvm是pe的整数倍lvm:逻辑卷 ...
- vue混入(mixins)
混入(mixins)是一种分发vue组件中可复用功能的非常灵活的方式.混入对象可以包含任意组件选项. 当组件使用混入对象时,所以混入对象的选项将被混入该组件本身选项,当组件和混入对象含有同名选项时,这 ...
- yii2下的路由重写
1.在config/main.php中添加配置信息 文件位置如图所示(图中展示的是backend下的config,也可以在fronted和common的config中添加) 配置的代码标准格式如下 ' ...
- java课后思考题(三)
1.以下代码为何无法通过编译?哪儿出错了? 因为在Foo类中已经有了一个Foo类的有参构造函数,所以Foo类中已经不默认Foo()的无参构造函数,所以在new Foo()时无法调用构造函数.所以在无法 ...
- element -validateField校验提示
<el-form :model="numberValidateForm" ref="numberValidateForm" :rules="ru ...
- SQL事务的四种隔离级别
1未提交读(Read uncommitted):完全不锁表,所以会出现脏数据.2提交读(Read committed):1.事务1中update才锁表,可以select到最新数据. 事务2select ...
- checkbox多选、全选js效果
//全选checkbox function allCheck() { //全选input var all = $("input[name='all']"); //全部的input ...
- 【Linux】Debian 8 设置命令行界面的文本颜色
平时我们操作的系统命令行界面文本默认黑底白字,有时候会看不惯这种全篇都是白色字符,这个时候可以通过改变PS1环境变量来改变文本颜色.我个人喜欢黑底绿字的搭配,以下是我个人的命令行界面样式: 注意:以下 ...