题目链接:http://lightoj.com/volume_showproblem.php?problem=1422

很简单的区间DP的入门题。一开始这题想了很久就是想不出来。直到做了后面几道区间DP回过来终于想明白了。

区间DP可以使用记忆化搜索和直接DP的方法写。

这题的状态转移方程:

dp[i][j]=min(1+dp[i+1][j],dp[i+1][k-1]+dp[k][j])   (    a[i]==a[k]   i<k<=j   )

注意初始化。

/*
 * Light OJ 1422 - Halloween Costumes
 * http://lightoj.com/volume_showproblem.php?problem=1422
 *  区间DP的思想。
 *  比如要求解i到j,dp[i][j].
 *  就是考虑i的衣服,一种是i的衣服只有在i使用,那么就是dp[i+1][j]+1件
 *  然后再i+1~j中枚举k,如果a[i]==a[k].那么dp[i][j]=min(dp[i][j],dp[i+1][k-1]+dp[k][j])
 *  注意因为i的衣服是可以使用多次的,所以不需要加1,是dp[k][j]
 *  思想很妙
 */

#include <iostream>
#include <stdio.h>
#include <algorithm>
#include <string.h>
using namespace std;
const int MAXN=110;
int a[MAXN];
int dp[MAXN][MAXN];

int main()
{
    int T;
    int n;
    scanf("%d",&T);
    int iCase=0;
    while(T--)
    {
        iCase++;
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
            scanf("%d",&a[i]);
        memset(dp,0,sizeof(dp));
        for(int i=1;i<=n;i++)
            for(int j=i;j<=n;j++)
                dp[i][j]=j-i+1;
        for(int i=n-1;i>=1;i--)//注意DP的顺序
            for(int j=i+1;j<=n;j++)
            {
                dp[i][j]=dp[i+1][j]+1;//这个表示第i个的衣服在后面没有利用了
                for(int k=i;k<=j;k++)
                    if(a[i]==a[k])//用同一件
                        dp[i][j]=min(dp[i][j],dp[i+1][k-1]+dp[k][j]);
            }
        printf("Case %d: %d\n",iCase,dp[1][n]);
    }
    return 0;
}