我爱编程

算法学习之区间dp

2018-04-09  本文已影响200人  Steven1997

简介

区间dp,顾名思义就是在一段区间上进行动态规划。对于每段区间,他们的最优值都是由几段更小区间的最优值得到,是分治思想的一种应用,将一个区间问题不断划分为更小的区间直至一个元素组成的区间,枚举他们的组合 ,求合并后的最优值。

算法结构

设F[i,j](1<=i<=j<=n)表示区间[i,j]内的数字相加的最小代价
每次用变量k(i<=k<=j-1)将区间分为[i,k]和[k+1,j]两段

For l:=2 to n do // 枚举区间长度
for i:=1 to n do // 枚举区间的左端点
begin
j:=i+l-1; // 计算区间的右端点,因为区间长度和左端点循环嵌套枚举,保证了[i,j]内的所有子区间都被枚举到
if j>n then break; // 保证了下标不越界
for k:= i to j-1 do // 状态转移,去推出 f[i,j]
f[i , j]= max{f[ i,k]+ f[k+1,j]+ w[i,j] }
end;

这个结构必须记好,这是区间动态规划的代码结构。

例题

石子合并

题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=737

题意:有N堆石子排成一排,每堆石子有一定的数量。现要将N堆石子并成为一堆。合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆。求出总的代价最小值。

分析:要求n个石子归并,我们根据dp的思想划分成子问题,先求出每两个合并的最小代价,然后每三个的最小代价,依次知道n个。
定义状态dp[i][j]为从第i个石子到第j个石子的合并最小代价。
那么dp[i][j] = min(dp[i][k] + dp[k+1][j])
那么我们就可以从小到大依次枚举让石子合并,直到所有的石子都合并。
这个问题可以用到平行四边形优化,用一个s[i][j]=k 表示区间 i---j 从k点分开才是最优的,这样的话我们就可以优化掉一层复杂度,变为O(n^2)

代码1(无优化)

   #include <cstdio>
   #include <cstring>
   #include <algorithm>
   #define N 210
   int dp[N][N],sum[N];
   int main()
   {
       int n;
       while(~scanf("%d",&n))
       {
           int a[N];sum[0]=0;
           for(int i=1;i<=n;i++){
               scanf("%d",&a[i]);
               sum[i]=sum[i-1]+a[i];//因为要求解区间和,先维护前缀和
           }
           memset(dp,0,sizeof(dp));
           int i,j,l,k;
           for(l = 2; l <= n; ++l)//枚举区间长度
           {
               for(i = 1; i <= n - l + 1; ++i)//枚举区间左端点
               {
                   j = i + l - 1;//根据左端点和区间长度求区间右端点
                   if(j > n)
                   break;
                   dp[i][j] = 0x3f3f3f3f;
                   for(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;
   }

代码2(平行四边形优化)

   #include <cstdio>
   #include <cstring>
   #include <algorithm>
   #define N 210
   int dp[N][N],sum[N],s[N][N];
   int main()
   {
       int n;
       while(~scanf("%d",&n))
       {
           int a[N];sum[0]=0;
           memset(s,0,sizeof(s));
           for(int i=1;i<=n;i++){
               scanf("%d",&a[i]);
               s[i][i]=i;
               sum[i]=sum[i-1]+a[i];//因为要求解区间和,先维护前缀和
           }
           memset(dp,0,sizeof(dp));
           int i,j,l,k;
           for(l = 2; l <= n; ++l)//枚举区间长度
           {
               for(i = 1; i <= n - l + 1; ++i) //枚举区间左端点
               {
                   j = i + l - 1;//根据左端点和区间长度求区间右端点
                   if(j > n)
                   break;
                   dp[i][j] = 0x3f3f3f3f;
                   for(k = s[i][j-1]; k <= s[i+1][j]; ++k)//四边形优化
                   {
                       if(dp[i][j]>dp[i][k] + dp[k + 1][j] + sum[j] - sum[i-1])
                            {
                                   dp[i][j]=dp[i][k] + dp[k + 1][j] + sum[j] - sum[i-1];
                                   s[i][j]=k;
                            }
                   }
               }
           }
           printf("%d\n", dp[1][n]);
       }
       return 0;
   }

括号匹配

题目链接:http://poj.org/problem?id=2955

题意:给出一串的只有‘(’ ‘)’ '[‘ ']'四种括号组成的串,让你求解需要最少添加括号数让串中的所有括号完全匹配。

分析:
定义dp [ i ] [ j ] 为串中第 i 个到第 j 个括号的最大匹配数目
1.如果第 i 个和第 j 个匹配,则dp [ i ] [ j ] = dp [ i+1 ] [ j-1 ] + 2 ;
2.如果第 i 个和第 j 个不匹配,枚举中间分割点k(i <= k < j)
dp[ i ] [ j ] = max ( dp [ i ] [ j ] , dp [ i ] [ k ] + dp [ k+1 ] [ j ] )

代码

    #include <iostream>
    #include <cstring>
    #include <algorithm>
    #include <string>
    using namespace std;
    const int  N = 120;
    int dp[N][N];

    int main()
    {
        string s;
        while(cin>>s)
        {
            if(s=="end")
                break;
            memset(dp,0,sizeof(dp));
            int n = s.size();
            for(int len = 2;len <= n;len++)//枚举区间长度
            {
                for(int i = 0;i <= n - len; i++)//枚举区间左端点
                {
                    int j = i + len - 1;//确定区间右端点
                    if(j > n)
                    break;
                    if(s[i]=='('&&s[j]==')' || s[i]=='['&&s[j]==']')
                        dp[i][j]=dp[i+1][j-1]+2;
                    for(int k=i;k<j;k++)
                        dp[i][j]=max(dp[i][j],dp[i][k]+dp[k+1][j]);//枚举中间位置,注意j不取等号
                }
            }
            cout<<dp[0][n-1]<<endl;
        }
        return 0;
    }

如果要求打印路径,即输出匹配后的括号

题目链接: http://poj.org/problem?id=1141

代码:

    #include <string>
    #include <iostream>
    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    using namespace std;
    const int  N = 120;
    int dp[N][N],pos[N][N];   /*定义pos【i】【j】表示 i 到 j 从哪儿分开使得匹配添加括号最少,如果i和j匹配我们可以让pos【i】【j】=-1;*/
    string s;
    void show(int i,int j)
    {
        if(i>j)  return;
        if(i==j)
        {
            if(s[i]=='('||s[i]==')') cout<<"()";
            else      cout<<"[]";
        }
        else
        {
            if(pos[i][j]==-1)
            {
                cout<<s[i];
                show(i+1,j-1);
                cout<<s[j];
            }
            else
            {
                show(i,pos[i][j]);
                show(pos[i][j]+1,j);
            }
        }
    }
    int main()
    {

        while(cin>>s)
        {
            memset(dp,0,sizeof(dp));
            int len=s.size();
            for(int i=1; i<len; i++)
            {
                for(int j=0,k=i; k<len; j++,k++)
                {
                    if(s[j]=='('&&s[k]==')' || s[j]=='['&&s[k]==']')
                    {
                        dp[j][k]=dp[j+1][k-1]+2;
                        pos[j][k]=-1;
                    }
                    for(int f=j; f<k; f++)
                    {
                        if(dp[j][f]+dp[f+1][k]>=dp[j][k])
                        {
                            dp[j][k]=dp[j][f]+dp[f+1][k];
                            pos[j][k]=f;
                        }
                    }
                }
            }

            show(0,len-1);
            cout<<endl;
        }
        return 0;
    }

整数划分

题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=746

题意: 给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积

分析: 区间dp,设dp[i][j] 表示在区间[0, i]之中,插入j个乘号可以得到的最大数
设a[i][j]为区间[i,j]所形成的数
所以 dp[i][j] = max(dp[k][j-1] * a[k + 1][i])

代码:

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

    using namespace std;

    long long dp[25][25];
    long long a[25][25];
    char str[25];

    int main()
    {
        int len, t, m;
        scanf("%d", &t);
        while (t--)
        {
            scanf("%s%d", str, &m);
            len = strlen(str);
            m--;
            memset (a, 0, sizeof(a));
            memset (dp, 0, sizeof(dp));
            for (int i = 0; i < len; i++)          //先对a进行预处理,减少复杂度,a[i][j]表示第i段到第j段的数值
            {
                a[i][i] = str[i] - '0';
                for (int j = i + 1; j < len; j++)
                {
                    a[i][j] = a[i][j - 1] * 10 + str[j] - '0';
                }
            }
            for (int i = 0; i < len; i++)
            {
                dp[i][0] = a[0][i];
            }
            for (int j = 1; j <= m; j++)
            {
                for (int i = j; i < len; i++)
                {
                    for (int k = 0; k < i; k++)
                    {
                        dp[i][j] = max(dp[i][j], dp[k][j - 1] * a[k + 1][i]);
                    }
                }
            }
            printf("%lld\n", dp[len - 1][m]);
        }
        return 0;
    }

Halloween Costumes

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

题意:给你n天需要穿的衣服的样式,每次可以套着穿衣服,脱掉的衣服就不能再穿了,问至少要带多少条衣服才能参加所有宴会

分析:首先我们使用dp[a][b]来表示区间 a~b 的答案,那么对于第 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<cstdio>
    #include<algorithm>

    using namespace std;

    int n;
    int a[105];
    int dp[105][105];

    int main(void)
    {
        int t;
        int cas = 0;
        scanf("%d",&t);
        while(t--)
        {
            cas ++;
            scanf("%d",&n);
            for(int i = 1;i <= n;i++) scanf("%d",&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]);
                        }
                    }
                }
            }

            printf("Case %d: %d\n",cas,dp[1][n]);
        }
        return 0;
    }

Cheapest Palindrome

题目链接:http://poj.org/problem?id=3280

题意:给你m个字符,其中有n种字符,每种字符都有两个值,分别是增加一个这样的字符的代价,删除一个这样的字符的代价,让你求将原先给出的那串字符变成回文串的最小代价。

分析:dp[i][j]代表区间i到区间j成为回文串的最小代价,那么对于dp[i][j]有三种情况:

1、dp[i+1][j]表示区间i到区间j已经是回文串了的最小代价,那么对于s[i]这个字母,我们有两种操作,删除与添加,对应有两种代价,dp[i+1][j]+add[s[i]],dp[i+1][j]+del[s[i]],取这两种代价的最小值;

2、dp[i][j-1]表示区间i到区间j-1已经是回文串了的最小代价,那么对于s[j]这个字母,同样有两种操作,dp[i][j-1]+add[s[j]],dp[i][j-1]+del[s[j]],取最小值

3、若是s[i]==s[j],dp[i+1][j-1]表示区间i+1到区间j-1已经是回文串的最小代价,那么对于这种情况,我们考虑dp[i][j]与dp[i+1][j-1]的大小

然后dp[i][j]取上面这些情况的最小值

代码

    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    int dp[2005][2005],add[27],del[27];
    char s[2005];

    int main()
    {
        int n,m;
        while(scanf("%d%d",&n,&m)>0)
        {
            scanf("%s",s+1);
            for(int i=1;i<=n;i++)
            {
                char ch[10];
                int tmp1,tmp2;
                scanf("%s%d%d",ch,&tmp1,&tmp2);
                add[ch[0]-'a'+1]=tmp1;
                del[ch[0]-'a'+1]=tmp2;
            }
            memset(dp,0,sizeof(dp));
            for(int i=m-1;i>=1;i--)
            {
                for(int j=i+1;j<=m;j++)
                {
                    dp[i][j]=min(dp[i+1][j]+add[s[i]-'a'+1],dp[i+1][j]+del[s[i]-'a'+1]);
                    int tmp=min(dp[i][j-1]+add[s[j]-'a'+1],dp[i][j-1]+del[s[j]-'a'+1]);
                    dp[i][j]=min(dp[i][j],tmp);
                    if(s[i]==s[j])
                    dp[i][j]=min(dp[i][j],dp[i+1][j-1]);
                }
            }
            printf("%d\n",dp[1][m]);
        }
        return 0;
    }

Treats for the Cows

题目链接:http://poj.org/problem?id=3186

题意:只能从一个序列的左右两端取数字,且取出的第i个数乘i,求乘积相加的最大值

分析:设dp[i][j]为取到剩余区间为[i,j]的最大值,可能由d[i+1][j]或者d[i][j-1]转移而来
转移方程:dp[i][j]=max(dp[i+1][j]+p[i](n+i-j),dp[i][j-1]+p[j](n+i-j)); 其中n-(j-i)是第几次取

代码

    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    using namespace std;
    int p[2010];
    int dp[2010][2010];
    int n;

    int main()
    {
        while(scanf("%d",&n)!=EOF)
        {
            memset(dp,0,sizeof(dp));
            for(int i=1;i<=n;i++)
            {
                scanf("%d",&p[i]);
                dp[i][i]= n * p[i];
            }
            int ans=0;
            for(int i=n;i>=1;i--)
                for(int j=i;j<=n;j++)
            {
               dp[i][j]=max(dp[i+1][j]+p[i]*(n+i-j),dp[i][j-1]+p[j]*(n+i-j));
            }
            printf("%d\n",dp[1][n]);
        }
    }
上一篇 下一篇

猜你喜欢

热点阅读