PS:如果读过题了可以跳过题目描述直接到题解部分

提交链接:洛谷 P2331 [SCOI2005]最大子矩阵

题目

题目描述

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

输入格式

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

输出格式

只有一行为 k 个子矩阵分值之和最大为多少。

样例

样例输入

3 2 2
1 -3
2 3
-2 3

样例输出

9

题解

只有一列

对于只有一列的情况,我们只需要考虑三种情况,首先是当前数字选还是不选,选的话是与上一个子矩阵合并还是重新开启一个子矩阵。

因此,我们可以用一个三维数组来进行状态转移:dp[i][j][0/1]表示前 i 行有 j 个子矩阵且当前行选(1)或不选(0)。

不选的话,无论上一行是什么情况其实都不影响,所以可以直接转移:dp[i][j][0]=max(dp[i-1][j][0],dp[i-1][j][1])

选的话,如果重新开启一个子矩阵,那么就和上一行的状态没有关系了,为了代码写着方便一些,我直接用了不选的情况进行转移(值是一样的),另一种情况是与上一个子矩阵合并,所以转移方程是:dp[i][j][1]=max(dp[i][j][0],dp[i-1][j][1])+a[i][1]

变成两列

变成两列的时候,我们需要考虑每一行有五种不同的状态:

  1. 两列都不选(00)
  2. 只选右边一列(01)
  3. 只选左边一列(10)
  4. 左右两边都选并且作为一个整体(11)
  5. 左右两边都选但是分别选(11*)

这五种情况我们分别来看。

两列都不选(00):这和只有一列的情况差不多,都是和上一行的状态没有关系,所以需要全部转移。dp[i][j][0]=max(dp[i-1][j][0],max(dp[i-1][j][1],max(dp[i-1][j][2],max(dp[i-1][j][3],dp[i-1][j][4]))))

只选右边一列(01):既可以重新开启一个子矩阵(从当行的两列都不选转移),也可以与上一个子矩阵合并,合并的话有两种,一种是从(01)合并,但比较容易忽略掉的,也可以从(11*)合并,而且最后要记得加上本行选择了的值。dp[i][j][1]=max(dp[i][j-1][0],max(dp[i-1][j][1],dp[i-1][j][4]))+a[i][2]

只选左边一列(10):和只选右边一列的方法是一样的。dp[i][j][2]=max(dp[i][j-1][0],max(dp[i-1][j][2],dp[i-1][j][4]))+a[i][1]

左右两边都选并且作为一个整体(11):这种状态比较特殊,因为它只能要么重新开启一个子矩阵,要么就只能从上一行相同的状态转移。很需要注意的是,它不能从(11*)转移。大体和前面其实也是差不多的。dp[i][j][3]=max(dp[i][j-1][0],dp[i-1][j][3])+a[i][1]+a[i][2]

左右两边都选但是分别选(11*):这种状态可以从(00)、(01)、(10)、(11*)四种状态转移,但需要注意每种转移状态的矩阵的减少量不同。dp[i][j][4]=max(dp[i][j-2][0],max(dp[i-1][j-1][1],max(dp[i-1][j-1][2],dp[i-1][j][4])))+a[i][1]+a[i][2]

总结

这道题其实想明白了就很简单,最难的地方在于区分每一行的状态(虽然我被卡住的地方是如何定义状态),只要想明白了每一行的状态是如何区分的,再注意一下取值范围,该加特判的地方加特判,这道题也就解决了。

代码实现

//洛谷 P2331 [SCOI2005]最大子矩阵
#include<iostream>
#include<cstdio>
using namespace std;
int n,m,k;
int a[110][5];
int dp[110][20][10];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){//只有一列for(int i=1;i<=n;++i){for(int j=0;j<=k;++j){dp[i][j][0]=max(dp[i-1][j][0],dp[i-1][j][1]);//不选if(j){dp[i][j][1]=max(dp[i][j-1][0],dp[i-1][j][1])+a[i][1];//选}}}printf("%d\n",max(dp[n][k][0],dp[n][k][1]));}else{//变成两列for(int i=1;i<=n;++i){for(int j=0;j<=k;++j){dp[i][j][0]=max(dp[i-1][j][0],max(dp[i-1][j][1],max(dp[i-1][j][2],max(dp[i-1][j][3],dp[i-1][j][4]))));if(j){dp[i][j][1]=max(dp[i][j-1][0],max(dp[i-1][j][1],dp[i-1][j][4]))+a[i][2];dp[i][j][2]=max(dp[i][j-1][0],max(dp[i-1][j][2],dp[i-1][j][4]))+a[i][1];dp[i][j][3]=max(dp[i][j-1][0],dp[i-1][j][3])+a[i][1]+a[i][2];if(j>=2){dp[i][j][4]=max(dp[i][j-2][0],max(dp[i-1][j-1][1],max(dp[i-1][j-1][2],dp[i-1][j][4])))+a[i][1]+a[i][2];}} }}printf("%d\n",max(dp[n][k][0],max(dp[n][k][1],max(dp[n][k][2],max(dp[n][k][3],dp[n][k][4])))));}return 0;
}

洛谷 P2331 [SCOI2005]最大子矩阵相关推荐

  1. [T][3]洛谷 P2331 [SCOI2005] 最大子矩阵

    题目描述 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 输入输出格式 输入格式: 第一行为n,m,k(1≤n≤100,1≤m≤2 ...

  2. BZOJ1084洛谷P2331 [SCOI2005]最大子矩阵

    DP+思维 思路 这道题的切入点是mmm,发现mmm只有两种取值,那么我们就可以尝试对mmm分类讨论 m=1 发现在m=1m=1m=1时就是在一个一维序列上做k个最大子段和,我们定义f[i][j]f[ ...

  3. 信息学奥赛一本通 1392:繁忙的都市(city) | 洛谷 P2330 [SCOI2005]繁忙的都市

    [题目链接] ybt 1392:繁忙的都市(city) 洛谷 P2330 [SCOI2005]繁忙的都市 [题目考点] 1. 图论 最小生成树 [解题思路] 将题目叙述转为图论概念,交叉路口为顶点,道 ...

  4. 洛谷 P2327 [SCOI2005]扫雷

    PS:如果读过题了可以跳过题目描述直接到题解部分 提交链接:洛谷 P2327 [SCOI2005]扫雷 题目 题目描述 相信大家都玩过扫雷的游戏.那是在一个 n*m 的矩阵里面有一些雷,要你根据一些信 ...

  5. P2331 [SCOI2005]最大子矩阵 题解

    DP 题,好像有点恶心,主要是因为能不能选空矩阵的问题. 有些数据好像是可以选空矩阵的有些又不能选,就很离谱,但是根据原数据来看空矩阵应该是不能选的,我也不知道具体情况() 注意到 m=1m=1m=1 ...

  6. P2331 [SCOI2005]最大子矩阵(DP分类讨论)

    P2331 [SCOI2005]最大子矩阵(DP&分类讨论) 考虑dp解决. 若m=1m=1m=1则是一个简单的递推. 若m=2m=2m=2则是分情况讨论. 一个是当前行不取. 一个是选第一列 ...

  7. 洛谷P2327 [SCOI2005]扫雷 题解

    [SCOI2005]扫雷 - 洛谷 description: 一个 的棋盘.已知右侧一列全部没有雷,且已知第 行相应的格子为 ,表示八联通的格子内共有 个雷.求左侧一列可能的雷的方案数. soluti ...

  8. 洛谷P2331最大子矩阵

    其实那,我是被标题吸引来的 你康康,最大子矩阵!多么人畜无害的名字啊~~~ 然后发生了什么吗大家都应该猜到啦!然后一读题,让你求出k个子矩阵的最大值! 但是呢?发现这是一个n*m的矩阵废话!然后.. ...

  9. [dp]洛谷 P2331 最大子矩阵

    题目描述 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 输入输出格式 输入格式: 第一行为n,m,k(1≤n≤100,1≤m≤2 ...

最新文章

  1. 2018-3-20论文(一种新型的智能算法-狼群算法WPA)笔记二(狼群系统分析,算法步骤)
  2. php双向链表,双向链表的GO语言实现
  3. 数据结构笔记 递推与迭代
  4. 3DSlicer25:Report an Error
  5. 机械制图国家标准的绘图模板_如何使用p5js构建绘图应用
  6. numpy维度交换_数据分析-gt;基本操作numpy(1)
  7. 韩顺平php视频笔记77 抽象类vs接口 关键字final const
  8. TCP/IP基础介绍
  9. 蓝桥杯省赛JavaB组真题
  10. java错误代码1061_求助java大神,看下这是哪里出错了
  11. CRM系统源码PHP开发
  12. 三村合建水厂问题研究
  13. Linux C 函数指针应用---回调函数
  14. 拜尔滤色拜尔滤色镜_如何在Windows 10上启用滤色器以更清晰地阅读屏幕
  15. 密码应用安全性评估实施要点之二密码技术应用要求与实现要点(4)
  16. 超动感,百行Python代码制作动态气泡图
  17. 极限交付:软件项目外包成功的保障
  18. 模拟电子技术基础 第一章 绪论
  19. brpc源码分析——数据报处理过程
  20. 用户运营指标体系建设实践 by 千冰仪

热门文章

  1. html语言h1h2h3,什么是H标签?H1,H2,H3标签使用的方法及重要性
  2. 数据库中级教程:第三讲 数据探查
  3. 归并排序、快速排序、二路快排、三路快排python实现
  4. 爆单助手教你各种型号打印机校验方法
  5. python 读取excel表格某列数据
  6. 计算机专业知识内容,计算机基础知识与基本操作
  7. uniapp组件和HTML标签重名致设置标签样式不生效
  8. 微型计算机系统中微处理器又称为什么,微处理器又称为什么
  9. 用20块的摄像头(不带fifo的OV7670)做WiFi实时传图小车
  10. 人工智能产品经理:人机对话系统设计逻辑探究(笔记)