zoukankan      html  css  js  c++  java
  • BZOJ 1084 最大子矩阵 dp

    题目链接:

    https://www.lydsy.com/JudgeOnline/problem.php?id=1084

    题目大意:

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

    思路:

    m=1时,就是数组选出k个连续子段和最大。

    dp[i][j]表示前i个数中已经取了j个连续子段和的最优解。

    dp[i][j] = dp[i-1][j] 不取这个数

    dp[i][j] = dp[start-1][j-1] + s[start]+...+s[i] 取这个数,并且从start到i作为第j个连续段(求区间和直接用前缀和求差代替)

    m=2时

    dp[i][j][k]表示第一列前i个数 第2列前j个数,已经取了k个子矩阵的最优解

    dp[i][j][k] = max(dp[i-1][j][k], dp[i][j-1][k]) 不取这个数

    dp[i][j][k] = dp[start-1][j][k-1] + s[start][1]+...+s[i][1] 第一列从start到i取出来作为第k个子矩阵

    dp[i][j][k] = dp[i][start-1][k-1] + s[start][2]+...+s[i][2] 第二列从start到i取出来作为第k个子矩阵

    dp[i][j][k] = dp[start-1][start-1][k-1] + s[start][1]+...+s[i][1]+s[start][2]+...+s[j][2]当且仅当i==j 两列同时取。

    同样的,区间求和用前缀和快速求出。

    时间复杂度为O(n^3*k)

     1 #include<bits/stdc++.h>
     2 #define IOS ios::sync_with_stdio(false);//不可再使用scanf printf
     3 #define Max(a, b) ((a) > (b) ? (a) : (b))//禁用于函数,会超时
     4 #define Min(a, b) ((a) < (b) ? (a) : (b))
     5 #define Mem(a) memset(a, 0, sizeof(a))
     6 #define Dis(x, y, x1, y1) ((x - x1) * (x - x1) + (y - y1) * (y - y1))
     7 #define MID(l, r) ((l) + ((r) - (l)) / 2)
     8 #define lson ((o)<<1)
     9 #define rson ((o)<<1|1)
    10 #define Accepted 0
    11 #pragma comment(linker, "/STACK:102400000,102400000")//栈外挂
    12 using namespace std;
    13 inline int read()
    14 {
    15     int x=0,f=1;char ch=getchar();
    16     while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();}
    17     while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    18     return x*f;
    19 }
    20 
    21 typedef long long ll;
    22 const int maxn = 100 + 10;
    23 const int MOD = 1000000007;//const引用更快,宏定义也更快
    24 const int INF = 1e9 + 7;
    25 const double eps = 1e-6;
    26 int dp1[maxn][11];//m = 1       dp1[i][j]表示前i个数中取了j个连续段的最大值
    27 int dp2[maxn][maxn][11];//m = 2     dp2[i][j][k]表示第一列前i个数第二列前j个数取了k个连续段的最大值
    28 int s[maxn];
    29 int sum[maxn][3];
    30 int main()
    31 {
    32     int n, m, K;
    33     scanf("%d%d%d", &n, &m, &K);
    34     if(m == 1)
    35     {
    36         for(int i = 1; i <= n; i++)scanf("%d", &s[i]), s[i] += s[i - 1];//直接记录前缀和
    37         for(int i = 0; i <= n; i++)
    38             for(int j = 1; j <= K; j++)dp1[i][j] = -INF;//dp[i][0]均为初始化的0
    39         for(int i = 1; i <= n; i++)
    40         {
    41             for(int j = 1; j <= K; j++)
    42             {
    43                 dp1[i][j] = dp1[i - 1][j];//不取这个数字
    44                 for(int start = 1; start <= i; start++)//从start开始一直取到i作为第j段
    45                 {
    46                     dp1[i][j] = max(dp1[i][j], dp1[start - 1][j - 1] + s[i] - s[start - 1]);
    47                 }
    48             }
    49         }
    50         printf("%d
    ", dp1[n][K]);
    51     }
    52     else
    53     {
    54         for(int i = 1; i <= n; i++)
    55         {
    56             for(int j = 1; j <= m; j++)
    57             {
    58                 scanf("%d", &sum[i][j]);
    59                 sum[i][j] += sum[i - 1][j];//记录每一列的前缀和
    60             }
    61         }
    62         for(int i = 0; i <= n; i++)for(int j = 0; j <= n; j++)for(int k = 1; k <= K; k++)dp2[i][j][k] = -INF;
    63         for(int i = 1; i <= n; i++)
    64             for(int j = 1; j <= n; j++)for(int k = 1; k <= K; k++)
    65         {
    66             dp2[i][j][k] = max(dp2[i - 1][j][k], dp2[i][j - 1][k]);//不取这个数字
    67             for(int start = 1; start <= i; start++)//第一列从start开始一直取到i作为第k个矩阵
    68                 dp2[i][j][k] = max(dp2[i][j][k], dp2[start - 1][j][k - 1] + sum[i][1] - sum[start - 1][1]);
    69             for(int start = 1; start <= j; start++)//第二列从start开始一直取到j作为第k个矩阵
    70                 dp2[i][j][k] = max(dp2[i][j][k], dp2[i][start - 1][k - 1] + sum[j][2] - sum[start - 1][2]);
    71             if(i == j)//两列从start开始一直取到i作为第k个矩阵
    72             {
    73                 for(int start = 1; start <= i; start++)
    74                     dp2[i][j][k] = max(dp2[i][j][k], dp2[start - 1][start - 1][k - 1] + sum[i][1] + sum[i][2] - sum[start - 1][1] - sum[start - 1][2]);
    75             }
    76         }
    77         printf("%d
    ", dp2[n][n][K]);
    78     }
    79     return Accepted;
    80 }
  • 相关阅读:
    烟大课表PC端-不仅仅是浏览器和手机APP
    关于51单片机电子时钟精度的问题
    第十二周项目4-点、圆的关系
    Git on Windows 一些问题
    vi 的使用
    Git 账户认证的一些问题
    [Windows] win7 配置Java开发环境
    Velocity 局部定制模板
    [Storm] Storm与asm的恩恩怨怨
    [Storm] No data flows into bolt
  • 原文地址:https://www.cnblogs.com/fzl194/p/9683511.html
Copyright © 2011-2022 走看看