Codeforces 1114D Flood Fill (区间DP or 最长公共子序列)
题意:给你n个颜色块,颜色相同并且相邻的颜色块是互相连通的(连通块)。你可以改变其中的某个颜色块的颜色,不过每次改变会把它所在的连通块的颜色也改变,问最少需要多少次操作,使得n个颜色块的颜色相同。
例如:[1, 2, 2, 3, 2]需要2步:[1, 2, 2, 3, 2] -> [1, 2, 2, 2, 2] -> [2, 2, 2, 2, 2]。
思路:我们先把颜色块压缩(即把本来颜色相同并且相邻的颜色块合并)。容易发现,如果出现了形如[1, 2, 3, 1]这种情况, 那么显然先合并两个1中间的部分,再把中间部分变成1操作次数会更少,于是我们就可以得到2种思路:
法1:区间DP:设dp[l][r]是合并[l, r]之间的颜色块的最小的代价,那么分两种情况:
1: l - 1和r + 1颜色相同,dp[l - 1][r + 1] = min(dp[l - 1][r + 1], dp[l][r] + 1);//l - 1和r + 1颜色相同,一步就可以转移到
2: l - 1和r + 1颜色不同,dp[l - 1][r] = min(dp[l - 1][r], dp[l][r] + 1);dp[l][r + 1] = min(dp[l][r + 1], dp[l][r] + 1);//l - 1和r + 1颜色不同,那只能一步转移到l - 1, r或者l, r + 1
代码:
#include <bits/stdc++.h>
using namespace std;
const int maxn = 5010;
int dp[maxn][maxn];
int a[maxn], b[maxn], tot;
int main() {
int n;
scanf("%d", &n);
for (int i = 1; i <= n; i++) {
scanf("%d", &a[i]);
}
for (int i = 1; i <= n; i++) {
if(a[i] == a[i + 1]) continue;
b[++tot] = a[i];
}
memset(dp, 0x3f, sizeof(dp));
for (int i = 1; i <= tot; i++) dp[i][i] = 0;
for (int len = 1; len <= tot; len++) {
for (int l = 1; l <= tot - len + 1; l++) {
int r = l + len - 1;
if(b[l - 1] == b[r + 1]) {
dp[l - 1][r + 1] = min(dp[l - 1][r + 1], dp[l][r] + 1);
}
else {
dp[l - 1][r] = min(dp[l - 1][r], dp[l][r] + 1);
dp[l][r + 1] = min(dp[l][r + 1], dp[l][r] + 1);
}
}
}
printf("%d\n", dp[1][tot]);
}
法2:假设n个颜色块的颜色都各不相同,那么合并需要n - 1次。如果出现了[1, 2, 3, 1]这种情况,那么合并次数可以少一次。如果是[4, 1, 2, 3, 4, 3, 2, 1]这种情况,那么先合并3到3之间,再合并2到2之间,再合并1到1之间比先合并4到4之间更优。如果我们把压缩序列上相同并且相距最近的两个数看成区间,每出现一个区间答案会减一,所以答案是n - 1 - 不相交的最多的区间的数目(区间在端点相交不算相交,包含关系不算相交)。比如[4, 1, 2, 3, 4, 3, 2, 1]这种情况,有4个区间。4:[1, 5] , 1: [2, 8], 2: [3, 7], 3: [4, 6]。显然,保留 [2, 8], [3, 7], [4, 6]三个区间最优,答案是8 - 1 - 3 = 4。那么怎么求不相交的最多的区间的数目呢?这个区间数目是压缩序列和反向压缩序列的的LCS / 2。例如[1, 2, 3, 1]和[1, 3, 2, 1]的LCS是3,区间个数是1。为什么呢?求lCS的过程中,我们每匹配成功一次,相当于区间的左端点找到的一个可以匹配的区间的右端点,或者是一个右端点找到一个对应的左端点,而匹配过程中是一个从前往后, 一个从后往前,所以匹配成功的区间肯定不相交。这个过程之后,每个区间被算了2次,并且会匹配出一个左端点等于右端点的区间,所以要除以2。
代码:
#include <bits/stdc++.h>
using namespace std;
const int maxn = 5010;
int dp[maxn][maxn];
int a[maxn], b[maxn], c[maxn], tot;
int main () {
int n;
scanf("%d", &n);
for (int i = 1; i <= n; i++) {
scanf("%d", &a[i]);
}
for (int i = 1; i <= n; i++) {
if(a[i] == a[i + 1]) continue;
b[++tot] = a[i];
c[tot] = a[i];
}
reverse(c + 1, c + 1 + tot);
for (int i = 1; i <= tot; i++)
for (int j = 1; j <= tot; j++) {
if(b[i] == c[j])
dp[i][j] = dp[i - 1][j - 1] + 1;
else
dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);
}
printf("%d\n", tot - 1 - dp[tot][tot] / 2);
}
Codeforces 1114D Flood Fill (区间DP or 最长公共子序列)的更多相关文章
- POJ-1458.CommonSubsequence.(DP:最长公共子序列裸题)
本题大意:给出两个字符串,让你求出最长公共子序列的长度并输出. 本题思路:本题是经典的DP问题,由于是两个字符串,那么我们就用一个二维数组来进行区分,用dp[ i ][ j ]来表示在s1和s2中分别 ...
- 51nod 1183 编辑距离【线性dp+类似最长公共子序列】
1183 编辑距离 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题 收藏 关注 编辑距离,又称Levenshtein距离(也叫做Edit Distance),是指两个 ...
- hdu 1080 dp(最长公共子序列变形)
题意: 输入俩个字符串,怎样变换使其所有字符对和最大.(字符只有'A','C','G','T','-') 其中每对字符对应的值如下: 怎样配使和最大呢. 比如: A G T G A T G - G ...
- dp(最长公共子序列)
A subsequence of a given sequence is the given sequence with some elements (possible none) left out. ...
- 1. 线性DP 1143. 最长公共子序列
最经典双串: 1143. 最长公共子序列 (LCS) https://leetcode-cn.com/problems/longest-common-subsequence/submissions/ ...
- D. Flood Fill 区间DP 或lcs匹配
题意 给定一串数字 相同的连续的数字可以同时 转换成一个相同数字 问最小几次可以全部转换成一个相同的数字 法1:区间dp dp[l][r][0/1] 0表示l r区间转化成和最左边相同需要多少次 ...
- codeforces1114D. Flood Fill(区间Dp)
传送门: 解题思路: 区间Dp,发现某一个区间修改后区间颜色一定为左边或右边的颜色. 那么只需要设方程$f_(l,r,0/1)$表示区间$[l,r]$染成左/右颜色的最小代价 转移就是枚举左右颜色就好 ...
- [dp]LCS最长公共子序列
https://www.51nod.com/tutorial/course.html#!courseId=4 复杂度:${\rm O}(nm)$ 转移方程: #include<bits/stdc ...
- POJ 1159 Palindrome(区间DP/最长公共子序列+滚动数组)
Palindrome Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 56150 Accepted: 19398 Desc ...
随机推荐
- HTTP通道
通过httptunnel技术进行入侵示例 httptunnel是通过HTTP通道来传输其他协议数据的工具软件,下载地址为:www.http-tunnel. com,目前最新版本3.0.5 工具/原料 ...
- python 实现一个TwoSum的例子
今天无意中看到一个题目,也不是很难,就想着用python实现以下: 题目是数组中的两个数相加等于输入的一个target,然后输出数组的下标. 比如: [1,2,3,4,5,6] target=7 返 ...
- SSM整合(spring、springMVC、mybatis)
需要用的包: 包括:spring的包.springMVC的包.mybatis的包.数据库驱动包.json相关的包 配置如下,首先是mybatis的配置 <?xml version="1 ...
- nyoj-130-相同的雪花(hash)
题目链接 /* Name:NYOJ-130-相同的雪花 Copyright: Author: Date: 2018/4/14 15:13:39 Description: 将雪花各个分支上的值加起来,h ...
- 20165210 Java第二次实验报告
20165210 实验二 Java面向对象程序设计 一.面向对象程序设计1--单元测试和TDD 实验要求 参考 http://www.cnblogs.com/rocedu/p/6371315.html ...
- UVALive - 3490 Generator (AC自动机+高斯消元dp)
初始有一个空串s,从前n个大写字母中不断随机取出一个字母添加到s的结尾,出现模式串t时停止,求停止时s的长度期望. 这道题解法不唯一,比较无脑的方法是对模式串t建一个单串AC自动机,设u为自动机上的一 ...
- [TopCoder12727]FoxAndCity
vjudge 题意 你有一张\(n\)点的无向图,每个点有一个点权\(w_i\).图中原来存在一些边,你可以任意给这张图加上一些边. 记点\(i\)到点\(1\)的距离为\(d_i\),你需要最小化\ ...
- 试用 Eagle 9.1
试用 Eagle 9.1 有推挤功能 原理图可以设置组装变体. 输出的 CAM 可以自定义,没有 Protel 那么死板. 保存的文件是 xml 文件,可以自由解析.
- Windbg内核调试之一: Vista Boot Config设置
Windbg进行内核调试,需要一些基本的技巧和设置,在这个系列文章中,我将使用Windbg过程中所遇到的一些问题和经验记录下来,算是对Kernel调试的一个总结,同时也是学习Windows系统内核的另 ...
- 第11篇 PSR-0 规范
Mandatory A fully-qualified namespace and class must have the following structure \<Vendor Name&g ...