【湖南师范大学2018年大学生程序设计竞赛新生赛 L】【HDOJ2476】【区间DP】
https://www.nowcoder.com/acm/contest/127/L
L 小小粉刷匠
题目描述
输入描述:
对于每一个案例,我们第一行包括两个整数n,k(1<=n<=100,1<=k<=50,k<n),表示墙的长度为n,刷子的长度为k。第二行输入n个整数,表示对于墙的每一段指定的颜色。
输出描述:
输出一个数,表示小名最少刷多少次。
输入例子:
3 3
1 2 1
输出例子:
2
-->
输入
3 3
1 2 1
输出
2
输入
5 4
5 4 3 3 4
输出
3
题意:给一个大小为n的数组SKT和一个大小为n的空数组,一次只能将空数组一个区间(区间大小为(1~k))的值设置为同一个数,问最少操作几次能将空数组改成该数组。
题解:见下面一题的题解。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int main() {
int dp[][];
int qwq[];
memset(dp,0x3f3f3f3f,sizeof(dp));
int n, k;
scanf("%d%d",&n,&k);
for (int i = ; i < n; i++) {
scanf("%d", &qwq[i]);
}
for (int i = n - ; i >= ; i--) {
for (int j = i; j < n; j++) {
if (i == j)dp[i][j] = ;
else dp[i][j] = dp[i + ][j]+;
for (int skt = i + ; skt < i + k&&skt<=j; skt++) {
if (qwq[skt] == qwq[i]) {
if (skt == j) {
dp[i][j] = min(dp[i][j], dp[i + ][j]);
}
else dp[i][j] = min(dp[i][j], dp[i + ][skt] + dp[skt + ][j]);
}
}
}
}
cout << dp[][n - ] << endl;
// cout << sizeof(int) * 24300000/1024<<"KB" << endl;
return ;
}
http://acm.hdu.edu.cn/showproblem.php?pid=2476
String painter
Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 6211 Accepted Submission(s): 2992
The first line contains string A.
The second line contains string B.
The length of both strings will not be greater than 100.
题意:给出两个串s1和s2,一次只能将一个区间刷一次,问最少几次能让s1=s2
题解:看到区间,又是找最少操作次数,想到区间DP。
首先简化一下问题:假如S1本身是一个空串,则问题等同于上一题,则可以使用dp(i,j)表示涂改 i~j 这个区间所需要的最少次数,在涂改位置i的时候,可以看出来如果S1串已经涂改过的字母中不含有与位置 i 要改变成的字母相同的字母,则有dp(i,j)=dp(i+1,j)+1,如果存在相同字母,则可以在那个位置涂改的时候顺便把位置i涂改,此时也就可以进行更新dp(i,j) = min(dp(i,j), dp(i + 1,k) + dp(k + 1,j));答案就是dp(0,n-1),这里也就是上一题的解法了。
而这题的S1不是空串,这和是空串有什么不一样的呢,S1不是空串,那么就有可能S1(i)==S2(i)那么这个时候不需要涂改,也就是ans(i)=ans(i-1),这也是唯一的区别,否则就使用由空串得到的dp数组进行更新,ans(i)=min(ans(i),ans(j)+dp(j+1,i))(其中0<=j&&j<=i)
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int main() {
int dp[][];
int ans[];
char s1[], s2[];
while (scanf("%s%s", s1, s2) != EOF) {
int len = strlen(s1);
memset(ans, 0x3f3f3f3f, sizeof(ans));
memset(dp, , sizeof(dp));//dp[i][j]当S1为空串时区间i~j的最小需要改动次数
for (int i = len - ; i >= ; i--) {//倒着来是因为更新的时候使用的dp[i+1][k],需要dp[i+1][k]已经是最优
for (int j = i; j < len; j++) {
if (i == j)
dp[i][j] = ;
else
dp[i][j] = dp[i + ][j] + ;//不考虑是否存在相等的字母时,直接在i的位置上涂改,在原来的次数上+1
for (int k = i + ; k <= j; k++) {
if (s2[i] == s2[k]) {//考虑存在字母已经被涂改成与i位置需要涂改的字母,这里i就可以随着那个k进行涂改
if (k == j) {
dp[i][j] = min(dp[i][j], dp[i + ][j]);
}
else {//之所以使用dp[i+1][k]+dp[k+1][j]而不是直接dp[i+1][j]是因为dp[i+1][k]的涂改可以保证多涂改一个i不影响dp[i+1][k]的最优性,而不能保证dp[i+1][j]的最优性
dp[i][j] = min(dp[i][j], dp[i + ][k] + dp[k + ][j]);
}
}
}
}
}
for (int i = ; i < len; i++) {
ans[i] = dp[][i];
}
if (s1[] == s2[])ans[] = ;
for (int i = ; i < len; i++) {
if (s1[i] == s2[i]) {//二者字母相同时,则不需要涂改
ans[i] = ans[i - ];
}
else {
for (int j = ; j < i; j++) {//二者字母不相同时,和S1是空串的状况一样..所以可以使用dp数组更新
ans[i] = min(ans[i], ans[j] + dp[j + ][i]);
}
}
}
cout << ans[len - ] << endl;
}
return ;
}
【湖南师范大学2018年大学生程序设计竞赛新生赛 L】【HDOJ2476】【区间DP】的更多相关文章
- 湖南师范大学2018年大学生程序设计竞赛新生赛 A 齐神和心美的游戏【hash】
[链接]:A [题意]:给你n个数的序列和k.判断是否可以三个数组成k(同一个数可以拿多次) [分析]:每个数vis记录一下.2层循环.两数之和不超过k以及剩下的数出现在序列中那么ok. [代码]: ...
- 2018中国大学生程序设计竞赛 - 网络选拔赛 1001 - Buy and Resell 【优先队列维护最小堆+贪心】
题目传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6438 Buy and Resell Time Limit: 2000/1000 MS (Java/O ...
- 2018中国大学生程序设计竞赛 - 网络选拔赛 1010 YJJ's Salesman 【离散化+树状数组维护区间最大值】
题目传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6447 YJJ's Salesman Time Limit: 4000/2000 MS (Java/O ...
- 2018中国大学生程序设计竞赛 - 网络选拔赛 1009 - Tree and Permutation 【dfs+树上两点距离和】
Tree and Permutation Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Oth ...
- hdu6489 2018 黑龙江省大学生程序设计竞赛j题
Problem Description Kayaking is playing a puzzle game containing n different blocks. He marks the bl ...
- HDU - 6440 Dream 2018中国大学生程序设计竞赛 - 网络选拔赛
给定的\(p\)是素数,要求给定一个加法运算表和乘法运算表,使\((m+n)^p = m^p +n^p(0 \leq m,n < p)\). 因为给定的p是素数,根据费马小定理得 \((m+n) ...
- 2018中国大学生程序设计竞赛 - 网络选拔赛 Dream hdu6440 Dream 给出一个(流氓)构造法
http://acm.hdu.edu.cn/showproblem.php?pid=6440 题意:让你重新定义任意一对数的乘法和加法结果(输出乘法口诀表和加法口诀表),使得m^p+n^p==(m+n ...
- hdu6444 2018中国大学生程序设计竞赛 - 网络选拔赛 1007 Neko's loop
Neko's loop Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Total S ...
- 2018中国大学生程序设计竞赛 - 网络选拔赛 Solution
A - Buy and Resell 题意:给出n个交易点,每次能够选择买或者卖,求获得最大利润 思路:维护两个优先队列,一个是卖,一个是替换,当价格差相同时,优先替换,因为次数要最少 #includ ...
随机推荐
- 【oracle】oracle REGEXP_SUBSTR分割字符串
REGEXP_SUBSTR的使用:通过REGEXP_SUBSTR进行数据分割: , LEVEL, 'i') AS STR FROM DUAL CONNECT BY LEVEL <= LENGTH ...
- 验证码 kaptcha 参数详解
Constant 描述 默认值 kaptcha.border 图片边框,合法值:yes , no yes kaptcha.border.color 边框颜色,合法值: r,g,b (and optio ...
- OOP⑺
1.多态和instanceof 都是去买东西,但是根据我们给别人金额的不同,得到不同的结果!!!! 生活中的多态! 操作是否一致? 一致! 都是买东西! 什么不一样?? 01.消费金额不一样 02.因 ...
- 數據庫ORACLE轉MYSQL存儲過程遇到的坑~(總結)
ORACLE數據庫轉MySQL數據庫遇到的坑 總結 最近在做Oracle轉mysql的工程,遇到的坑是真的多,尤其是存儲過程,以前都沒接觸過類似的知識,最近也差不多轉完了就總結一下.希望能幫到一些人( ...
- C语言转义字符'\'
body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...
- 戴尔poweredge r730服务器配置及系统安装详解教程
第一次给服务器安装的是ubantu系统: 首先我们开机进入小型BIOS设置一下RAID,或者进入服务器管理系统,在系统的BIOS中进行RAID设置: 开机后当看到出现< Ctrl > 时按 ...
- 学react的第一天
属性 class = className for = htmlFrom jsx语法被编译了,所以可以在return里写html标签 react的属性 constructor(props){ super ...
- PLY文件格式
一.PLY简介 PLY文件格式是Stanford大学开发的一套三维mesh模型数据格式,图形学领域内很多著名的模型数据,比如Stanford的三维扫描数据库(其中包括很多文章中会见到的Happy Bu ...
- 5.4 C++重载输入与输出操作符
参考:http://www.weixueyuan.net/view/6382.html 总结: 在C++中,系统已经对左移操作符“<<”和右移操作符“>>”分别进行了重载,使其 ...
- quartz自定义线程数
1.加载包 2.添加quartz.propertes 3.编写自己的任务类 4.添加自动任务配置 5.通过 quartzProperties 配置连接池 1.加载包 <dependency> ...