zoukankan      html  css  js  c++  java
  • BZOJ1084 [SCOI2005]最大子矩阵 动态规划

    欢迎访问~原文出处——博客园-zhouzhendong

    去博客园看该题解


    题目传送门 - BZOJ1084


    题意概括

      这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大。注意:选出的k个子矩阵不能相互重叠。

      输入:第一行为n,m,k(1≤n≤100,1≤m≤2,1≤k≤10),接下来n行描述矩阵每行中的每个元素的分值(每个元素的分值的绝对值不超过32767)。


    题解

      注意到1<=m<=2!

      如果m = 1 ,那么就是一个简单的线性dp。

      我们设dp[i][j]表示在前i个里面选出k个子矩阵的最大分值。

      那么分两种情况讨论:

      1.  什么都不干: dp[i][j] = max(dp[i][j], dp[i-1][j])

      2. 弄一个新的子矩阵: dp[i][j] = max(dp[i][j], dp[x][j - 1] + presum[i] - presum[x])  0<=x<i

      时间复杂度O(kn2)

      如果 m = 2 ,那么是一个稍微复杂一点的线性dp。

      我们设dp[i][j][x]表示在第一列的前i个和第二列的前j个里面选出x个子矩阵的最大分值。

      那么分几种情况进行讨论:

      1. 什么都不干: dp[i][j][x] = max(dp[i][j][x], dp[i - 1][j][x], dp[i][j - 1][x])

      2. 在第一列弄一个新的子矩阵: dp[i][j][x] = max(dp[i][j][x], dp[y][j][x - 1] + presum[i][1] - presum[y][1])  0<=y<i

      3. 在第二列弄一个新的子矩阵: dp[i][j][x] = max(dp[i][j][x], dp[i][y][x - 1] + presum[j][2] - presum[y][2])  0<=y<j

      4. 在第一、二列弄一个宽度为2的子矩阵: dp[i][j][x] = max(dp[i][j][x], dp[y][y][x - 1] + presum[i][1] - presum[y][1] + presum[j][2] - presum[y][2])  i = j 且 0<=y<i

      时间复杂度O(kn3)


    代码

    #include <cstring>
    #include <algorithm>
    #include <cstdlib>
    #include <cstdio>
    #include <cmath>
    using namespace std;
    const int N=100+5,M=5,K=10+5;
    const int Inf=1<<25;
    int n,m,k,a[N][M];
    void solve1(){
        int dp[N][K],presum[N];
        for (int i=0;i<N;i++)
            for (int j=0;j<K;j++)
                dp[i][j]=-Inf;
        presum[0]=0;
        for (int i=1;i<=n;i++)
            presum[i]=presum[i-1]+a[i][1];
        dp[0][0]=0;
        int ans=-Inf;
        for (int i=0;i<=n;i++)
            for (int j=0;j<=k;j++){
                if (!i&&!j)
                    continue;
                if (i)
                    dp[i][j]=dp[i-1][j];
                if (!j)
                    continue;
                for (int x=0;x<i;x++)
                    dp[i][j]=max(dp[i][j],dp[x][j-1]+presum[i]-presum[x]);
            }
        printf("%d",dp[n][k]);
    }
    void solve2(){
        int dp[N][N][K],presum[N][M];
        presum[0][1]=presum[0][2]=0;
        for (int i=1;i<=n;i++){
            presum[i][1]=presum[i-1][1]+a[i][1];
            presum[i][2]=presum[i-1][2]+a[i][2];
        }
        for (int i=0;i<N;i++)
            for (int j=0;j<N;j++)
                for (int x=0;x<K;x++)
                    dp[i][j][x]=-Inf;
        dp[0][0][0]=0;
        for (int i=0;i<=n;i++)
            for (int j=0;j<=n;j++)
                for (int x=0;x<=k;x++){
                    if (!i&&!j&&!x)
                        continue;
                    if (i&&j)
                        dp[i][j][x]=max(dp[i-1][j][x],dp[i][j-1][x]);
                    else if (i)
                        dp[i][j][x]=dp[i-1][j][x];
                    else if (j)
                        dp[i][j][x]=dp[i][j-1][x];
                    if (!x)
                        continue;
                    for (int y=0;y<i;y++)
                        dp[i][j][x]=max(dp[i][j][x],dp[y][j][x-1]+presum[i][1]-presum[y][1]);
                    for (int y=0;y<j;y++)
                        dp[i][j][x]=max(dp[i][j][x],dp[i][y][x-1]+presum[j][2]-presum[y][2]);
                    if (i==j)
                        for (int y=0;y<i;y++)
                            dp[i][j][x]=max(dp[i][j][x],dp[y][y][x-1]+presum[i][1]-presum[y][1]+presum[j][2]-presum[y][2]);
                }
        printf("%d",dp[n][n][k]);
    }
    int main(){
        scanf("%d%d%d",&n,&m,&k);
        for (int i=1;i<=n;i++)
            for (int j=1;j<=m;j++)
                scanf("%d",&a[i][j]);
        if (m==1)
            solve1();
        else
            solve2();
        return 0;
    }
  • 相关阅读:
    express中间件
    复习node中加载静态资源--用express+esj
    有关es6的模块化
    es6转码和package.json中的配置
    MySQL必知必会--使用子查询
    MySQL必知必会--分 组 数 据
    MySQL必知必会--汇 总 数 据
    mysql必知必会--使用数据处理函数
    拼凑可导的充分必要条件
    递推数列极限存在证明
  • 原文地址:https://www.cnblogs.com/zhouzhendong/p/BZOJ1084.html
Copyright © 2011-2022 走看看