关闭

区间dp

标签: 动态规划
174人阅读 评论(0) 收藏 举报
分类:

区间dp

题目链接:https://vjudge.net/contest/169127#problem/F

这是我做的第一个区间dp,个人觉得,区间dp就是根据区间的变化来反映整个区间的变化,通过一个个小区间的动归的递推,很容易得出最后的整个区间。

题意:告诉有n场晚会中需要穿的衣服,衣服是可以套在其他衣服外面的,告诉了序列顺序之后求出最少需要穿多少次衣服。

解题思路:使用dp[i][j]来表示区间 i~j的答案,那么对于第 i 件衣服,我们有
①:如果在之后的区间内都不再重复利用这件衣服,那么明显 dp[i][j] = dp[i+1][j] + 1
②:如果在之后的区间 i+1 ~ j 中存在一件衣服 k 是跟 i 一样的,那么我们便可以考虑是不是可以将i那件衣服在k这个地方重复利用,那么转移方程为 dp[i][j] = min(dp[i][j] , dp[i][k-1]+dp[k+1][j])
因为是从后面的区间往前面转,所以循环的时候应该是从后往前的

各位大佬有什么好的见解可以给我留评论,小弟在此谢过。。。

代码如下:

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <string>

using namespace std;
typedef long long ll;
//const ll inf=1e15;
const ll inf=0x3f3f3f3f3f3f3f;
int a[100+10];
int dp[110][110];
int main()
{
    int t;
    cin>>t;
    int kase=0;
    while(t--)
    {
        int n;
        cin>>n;//n件衣服
        for(int i=1;i<=n;i++)
            cin>>a[i];
        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--)
        {
            for(int j=i+1;j<=n;j++)
            {
                dp[i][j]=dp[i+1][j]+1;
                for(int k=i+1;k<=j;k++)
                {
                    if(a[i]==a[k])
                        dp[i][j]=min(dp[i][j],dp[i][k-1]+dp[k+1][j]);//注意这个的第k天的衣服被省略掉了,因为第k天的衣服和第i天一样
                }
            }
        }
        cout<<"Case "<<++kase<<": ";
        cout<<dp[1][n]<<endl;
    }
    return 0;
}

补充:51nod 1021

题意:
N堆石子摆成一条线。现要将石子有次序地合并成一堆。规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的代价。计算将N堆石子合并成一堆的最小代价。

例如: 1 2 3 4,有不少合并方法
1 2 3 4 => 3 3 4(3) => 6 4(9) => 10(19)
1 2 3 4 => 1 5 4(5) => 1 9(14) => 10(24)
1 2 3 4 => 1 2 7(7) => 3 7(10) => 10(20)

括号里面为总代价可以看出,第一种方法的代价最低,现在给出n堆石子的数量,计算最小合并代价。
Input
第1行:N(2 <= N <= 100)
第2 - N + 1:N堆石子的数量(1 <= Aii <= 10000)
Output
输出最小合并代价

代码:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <string>
#include <algorithm>
#include <set>
#include <map>
#include <stack>
#include <vector>
#include <queue>
#define ri(n) scanf("%d",&n)
#define oi(n) printf("%d\n",n)
#define rl(n) scanf("%lld",&n)
#define ol(n) printf("%lld\n",n)
#define rep(i,l,r) for(i=l;i<=r;i++)
#define rep1(i,l,r) for(i=l;i<r;i++)
using namespace std;
typedef long long ll;
const int inf=0x3f3f3f3f;
const int maxn=50000+500;
const int epg=10-8;
int a[100+10],sum[100+10];
int dp[100+10][100+10];
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++)
        sum[i]=sum[i-1]+a[i];
    memset(dp,inf,sizeof(dp));
    for(int i=1;i<=n;i++)
        dp[i][i]=0;
    for(int len=1;len<=n;len++)
    {
        for(int i=1;i+len-1<=n;i++)
        {
            int j=i+len-1;
            for(int k=i;k<j;k++)
                dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+1][j]+sum[j]-sum[i-1]);
        }
    }
    printf("%d\n",dp[1][n]);
    return 0;
}
1
0
查看评论

区间DP入门之 石子归并问题

题目描述:有N堆石子排成一排,每堆石子有一定的数量。现要将N堆石子并成为一堆。合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆。求出总的代价最小值。 输入 有多组测试数据,输入到文件结束。 每组测试数据第一行有一个整数n,表示有n堆石子...
  • wuxuanyi27
  • wuxuanyi27
  • 2016-04-21 21:40
  • 1027

区间DP的主要思路

区间动态规划
  • xuanandting
  • xuanandting
  • 2015-07-31 16:08
  • 3945

动态规划学习系列——区间DP(一)

学习一个算法,还是从题目开始比较好,我们就从一道经典例题开始: wikioi 1048 石子归并 Description 有n堆石子排成一列,每堆石子有一个重量w[i], 每次合并可以合并相邻的两堆石子,一次合并的代价为两堆石子的重量和w[i]+w[i+1]。问安排怎样的合并顺序,能够使得...
  • fuyukai
  • fuyukai
  • 2015-02-13 17:13
  • 917

区间dp模型(石子归并,括号匹配,整数划分)

区间dp顾名思义就是在一个区间上进行的一系列动态规划。对一些经典的区间dp总结在这里。 1) 石子归并问题 题目链接:http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=737 描述: 有N堆石子排成一排,每堆石子有一定的数量...
  • y990041769
  • y990041769
  • 2014-04-20 21:51
  • 16578

区间DP 入门经典三道题

NYOJ 737:http://acm.nyist.net/JudgeOnline/problem.php?pid=737 代码:#include <cstdio> #include <iostream> #include <cstring> #define s...
  • SolarDomo
  • SolarDomo
  • 2016-08-15 14:17
  • 513

区间DP总结

做了几题区间动态规划的题目,觉得区间动态规划的题目是有点难的。 一点一点说吧! 首先我觉得首先区间DP的应用要先想到回文串的,包括一个字符串的最长的非连续的回文串,一个字符串非连续的回文串的数目。因为回文串的特点对应的两端字符是相等的,所以状态是可以转移的,先看一道求一个字符串中回文串的数目: ...
  • Dacc123
  • Dacc123
  • 2016-04-07 11:33
  • 1686

poj 3537 Minimax Triangulation(区间dp 三角剖分)

题意:给你一个n个顶点多边形,有很多种方法对它进行三角剖分,求最大三角形的面积最小的情况是多少? 思路:典型的区间dp 三角剖分类型的题目。这类题目的转移方程和矩阵链乘的很像,但是区别在于矩阵链乘可以反应决策的过程 ,而三角剖分的则允许随意切割,在这样的情况下,有必要把决策的顺序规范化,使得在规范...
  • yexiaohhjk
  • yexiaohhjk
  • 2016-09-16 18:38
  • 337

POJ 2955 Brackets (区间dp 括号匹配)

POJ 2955 Brackets (区间dp 括号匹配)
  • Tc_To_Top
  • Tc_To_Top
  • 2015-07-06 00:42
  • 1416

区间DP(初步了解)

区间动态规划问题一般都是考虑,对于每段区间,他们的最优值都 是由几段更小区间的最优值得到,是分治思想的一种应用,将一个区间 问题不断划分更小的区间直至一个元素组成的区间,枚举他们的组合  ,求合并后的最优值。设F[i,j](1 最小区间F[i,i]=0(一个数字无法合并,∴代价...
  • qq_16255321
  • qq_16255321
  • 2014-11-02 17:14
  • 1062

区间dp小结

区间dp,顾名思义,就是在区间上dp,即把整个区间划分为一个个的小区间,在小区间内dp求出最优值,然后把这些小区间合并以后就是整个取件的最优值。 下面是一些比较经典的区间dp题目: 1.NYOJ石子合并:http://acm.nyist.net/JudgeOnline/problem.php?pid...
  • LYHVOYAGE
  • LYHVOYAGE
  • 2014-04-24 09:39
  • 4044
    个人资料
    • 访问:13884次
    • 积分:884
    • 等级:
    • 排名:千里之外
    • 原创:76篇
    • 转载:2篇
    • 译文:0篇
    • 评论:3条