[CQOI2007]涂色paint(BZOJ 1260)题解

题目描述

假设你有一条长度为5的木版,初始时没有涂过任何颜色。你希望把它的5个单位长度分别涂上红、绿、蓝、绿、红色,用一个长度为5的字符串表示这个目标:RGBGR。

每次你可以把一段连续的木版涂成一个给定的颜色,后涂的颜色覆盖先涂的颜色。例如第一次把木版涂成RRRRR,第二次涂成RGGGR,第三次涂成RGBGR,达到目标。

用尽量少的涂色次数达到目标。

输入输出格式

输入格式:

输入仅一行,包含一个长度为n的字符串,即涂色目标。字符串中的每个字符都是一个大写字母,不同的字母代表不同颜色,相同的字母代表相同颜色。

输出格式:

仅一行,包含一个数,即最少的涂色次数。


非常明显的一道区间DP

关于区间DP转https://www.cnblogs.com/lizitong/p/10014809.html

我们要将一个区间涂成给出的颜色。

这道题的难点在于其与我们的思维相逆。

我们主观上的涂色全部都是先涂底层,然后在逐渐在上面覆盖,而这道题的状态转移方程却不同。

首先状态,dp[i][j]表示[i]到[j]的最少次数。

注意清初值,由于是求最小,初值清成正无穷,每个单独的颜色都需要涂,所以清成1。

当我们枚举长度的时候,如果[i][j]颜色相同,岂不是我在涂[i][j-1]或者[i+1][j]顺带一笔就可以带过去?

这就是我所说的与我们印象相逆的地方。假设我们现实中一笔带过去,中间的颜色全部会变色,但是我们这里是一个逆向的过程,中间的颜色是建立在已经涂过的颜色基础上的。

仔细想想。

那如果[i][j]颜色不一样呢?

那我们就需要把这段区间分成两段涂,枚举k在区间[i][j]中,把区间分成两段,然后将次数相加。

上代码。欢迎大家在评论区留言讨论。

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<string>
using namespace std;
char c[55];
int dp[55][55];
int main()
{
    scanf("%s",c+1);
    int l = strlen(c+1);
    memset(dp,0x3f,sizeof(dp));    
    for(int i = 1;i<=l;i++)
    {
        dp[i][i] = 1;
    }
    for(int len = 2;len<=l;len++)
    {
        for(int i = 1;i+len-1<=l;i++)
        {
            int j = i+len-1;
            if(c[i]==c[j])
            {
                dp[i][j] = min(dp[i+1][j],dp[i][j-1]);
            }else
            {
                for(int k = i;k<j;k++)
                {
                    dp[i][j] = min(dp[i][j],dp[i][k]+dp[k+1][j]);
                }
            }    
        }
    }
    printf("%d",dp[1][l]);
    return 0;
    
}

 

posted @ 2018-11-26 15:52  lizitong  阅读(287)  评论(1编辑  收藏  举报