NC20242 [SCOI2005]最大子矩阵
题目
题目描述
这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大。
注意:选出的k个子矩阵 不能相互重叠。
输入描述
第一行为n,m,k(1 ≤ n ≤ 100,1 ≤ m ≤ 2,1 ≤ k ≤ 10),
接下来n行描述矩阵每行中的每个元素的分值(每个元素的分值的绝对值不超过32767)。
输出描述
只有一行为k个子矩阵分值之和最大为多少。
示例1
输入
3 2 2
1 -3
2 3
-2 3
输出
9
题解
知识点:线性dp。
发现 \(m=1\) 时,就是 \(k\) 串最大和。
这里解释一下 \(k\) 串最大和的做法,有三种状态设置:取第 \(i\) 个数,共取了 \(j\) 串;考虑到第 \(i\) 个数,共取了 \(j\) 串;考虑到第 \(i\) 个数,共取了 \(j\) 串,第 \(i\) 个数的状态为0/1(不取/取)。
状态转移方程是 \(dp[i][j] = \max(dp[u][j-1],dp[i-1][j]) + a[i],0 \leq u \leq i-1\) ,表示 \(a[i]\) 独立一串和 \(a[i]\) 和前面串起来取最优解,时间复杂度是 \(O(n^2k)\) ,通过前缀最大值优化可以到 \(O(nk)\) 。
状态转移方程是 \(dp[i][j] = \max (dp[u][j-1]+sum[i] - sum[u],dp[i-1][j]), 0 \leq u \leq i-1\) ,表示选 \([u+1 , i]\) 一串和不选优解,时间复杂度是 \(O(n^2k)\)。
状态转移方程是:
\[\begin{aligned}
dp[i][j][0] &= \max (dp[i-1][j][1],dp[i-1][j][0])\\
dp[i][j][1] &= \max(dp[i-1][j][1],dp[i-1][j-1][1],dp[i-1][j-1][0])
\end{aligned}
\]表示不选就在前面的情况取最大值,选就在前面选后串一起或者在前面选后独立成一串或者前面不选独立成串中取最大值。时间复杂度是 \(O(nk)\) 。
要注意的是如果不允许取空串需要赋值负无穷且 \(k=0\) 的初态为 \(0\),如果允许则默认 \(0\) 即可。
现在扩展到 \(m=2\) 。也有三种设置:取第一列的第 \(i\) 个数和第二列的第 \(j\) 个数,共取了 \(u\) 个矩阵;考虑到第一列的第 \(i\) 个数和第二列的第 \(j\) 个数,共取了 \(u\) 个矩阵;考虑到第 \(i\) 行,共取了 \(j\) 个矩阵,第 \(i\) 行状态是 0/1/2/3/4(都不取\取第一列\取第二列\都取但不同块\都取成一块)。
这里写的是第二种,实际上第一种和第二种相似。第三种复杂度是 \(O(nk)\) ,但写起来麻烦,但可用矩阵运算优化写法。
转移方程为:
\left \{
\begin{array}{l}
dp[v][j][k-1]+sum[i][1]-sum[v][1] &,0\leq v\leq i-1\\
dp[i][v][k-1]+sum[j][2]-sum[v][2] &,0\leq v \leq j-1\\
dp[v][v][k-1]+sum[i][1]-sum[v][1]+sum[j][2]-sum[v][2] &,i = j \and 0\leq v \leq i-1\\
dp[i-1][j][k] &,i\geq 1\\
dp[i][j-1][k] &,j\geq 1
\end{array}
\right.
\]
分别指:
- 选第一列 \([v+1,i]\) 作为矩阵。
- 选第二列 \([v+1,j]\) 作为矩阵。
- 在 \(i=j\) 下还能选两列 \([v+1,i]\) 作为矩阵。
- 不选第一列。
- 不选第二列。
这道题数据不太行,其他题解有说可以有空矩阵,我这里写的是包括空矩阵的,但实际上初始化负无穷,\(k=0\) 初态为 \(0\) 做不包含空矩阵的,也是对的。
时间复杂度 \(O(n^2k)\)
空间复杂度 \(O(n^2k)\)
代码
#include <bits/stdc++.h>
using namespace std;
int a[107][10], dp[107][107][17], sum[107][10];
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n, m, k;
cin >> n >> m >> k;
for (int i = 1;i <= n;i++)
for (int j = 1;j <= m;j++)
cin >> a[i][j], sum[i][j] = a[i][j] + sum[i - 1][j];
///不需要从0开始,因为如果从(0,j,k-1)转移到(i,j,k),则有(min(i,j),min(i,j),k-1)到(i,j,k)的转移
///而形如(l,l,k)的状态是可以从 (1,1,1) 开始推的,因此所有都可以从(1,1,1)开始
///感觉好奇怪,还是从(0,0,1)开始舒服
///再者可以选空矩阵,因此不需要初始化负无穷
for (int i = 0;i <= n;i++) {
for (int j = 0;j <= n;j++) {
for (int u = 1;u <= k;u++) {
dp[i][j][u] = max(dp[max(0, i - 1)][j][u], dp[i][max(0, j - 1)][u]);///不选i或j
for (int v = 0;v < i;v++) dp[i][j][u] = max(dp[v][j][u - 1] + sum[i][1] - sum[v][1], dp[i][j][u]);///选i
for (int v = 0;v < j;v++) dp[i][j][u] = max(dp[i][v][u - 1] + sum[j][2] - sum[v][2], dp[i][j][u]);///选j
if (i == j)
for (int v = 0;v < i;v++) dp[i][j][u] = max(dp[v][v][u - 1] + sum[i][1] - sum[v][1] + sum[j][2] - sum[v][2], dp[i][j][u]);///选i和j相连
}
}
}
cout << dp[n][n][k] << '\n';
return 0;
}
NC20242 [SCOI2005]最大子矩阵的更多相关文章
- BZOJ 1084: [SCOI2005]最大子矩阵 DP
1084: [SCOI2005]最大子矩阵 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1084 Description 这里有一个n* ...
- 1084: [SCOI2005]最大子矩阵
1084: [SCOI2005]最大子矩阵 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1325 Solved: 670[Submit][Stat ...
- bzoj千题计划198:bzoj1084: [SCOI2005]最大子矩阵
http://www.lydsy.com/JudgeOnline/problem.php?id=1084 m=1: dp[i][j] 前i个数,选了j个矩阵的最大和 第i个不选:由dp[i-1][j] ...
- 【BZOJ 1084】 1084: [SCOI2005]最大子矩阵 (DP)
1084: [SCOI2005]最大子矩阵 Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第 ...
- BZOJ(6) 1084: [SCOI2005]最大子矩阵
1084: [SCOI2005]最大子矩阵 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3566 Solved: 1785[Submit][Sta ...
- [Luogu 2331] [SCOI2005]最大子矩阵
[Luogu 2331] [SCOI2005]最大子矩阵 题目描述 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 输入输出格式 ...
- 洛谷P2331 [SCOI2005]最大子矩阵 DP
P2331 [SCOI2005]最大子矩阵 题意 : 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 第一行为n,m,k(1≤n≤ ...
- [bzoj1084][SCOI2005]最大子矩阵_动态规划_伪·轮廓线dp
最大子矩阵 bzoj-1084 SCOI-2005 题目大意:给定一个n*m的矩阵,请你选出k个互不重叠的子矩阵使得它们的权值和最大. 注释:$1\le n \le 100$,$1\le m\le 2 ...
- [SCOI2005]最大子矩阵
题目描述 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 输入输出格式 输入格式: 第一行为n,m,k(1≤n≤100,1≤m≤2 ...
随机推荐
- elasticsearch-spark的用法
Hadoop允许Elasticsearch在Spark中以两种方式使用:通过自2.1以来的原生RDD支持,或者通过自2.0以来的Map/Reduce桥接器.从5.0版本开始,elasticsearch ...
- 『忘了再学』Shell基础 — 19、使用declare命令声明变量类型
目录 1.declare命令介绍 2.声明数组变量类型 3.声明变量为环境变量 4.声明只读属性 5.补充: 1.declare命令介绍 Shell中所有变量的默认类型是字符串类型,如果你需要进行特殊 ...
- CF1682E Unordered Swaps
鸽着,我不知道为什么对? 题意: 思路: code: #include<bits/stdc++.h> using namespace std; const int N=5e5+5; int ...
- 原理:C++为什么一般把模板实现放入头文件
写在前面 本文通过实例分析与讲解,解释了为什么C++一般将模板实现放在头文件中.这主要与C/C++的编译机制以及C++模板的实现原理相关,详情见正文.同时,本文给出了不将模板实现放在头文件中的解决方案 ...
- 如何为Java面试准备项目经验
1 提出问题 应届生朋友或Java程序员在找Java方面的工作时,一定会需要准备Java项目经验,但事实上不少求职者,是没有项目经验,或者只具有开源社区等的学习项目经验,这样的话,就很有可能在面试时无 ...
- github新项目npm错误
当我们从GitHub或者别人那里拿到项目的时候,一般都是要先npm install 进行安装依赖.但是难免会遇到报错. 出现问题1: 解决方案:清除缓存npm cache clear --force之 ...
- Prometheus 四种metric类型
Prometheus的4种metrics(指标)类型: Counter Gauge Histogram Summary 四种指标类型的数据对象都是数字,如果要监控文本类的信息只能通过指标名称或者 la ...
- Jmeter(五十四) - 从入门到精通高级篇 - 如何在linux系统下运行jmeter脚本 - 上篇(详解教程)
1.简介 上一篇宏哥已经介绍了如何在Linux系统中安装Jmeter,想必各位小伙伴都已经在Linux服务器或者虚拟机上已经实践并且都已经成功安装好了,那么今天宏哥就来介绍一下如何在Linux系统下运 ...
- docker 操作 记录
docker ps #查看当前docker容器 docker exec -it 容器名称 sh 进入docker容器 docker stop 停止docker容器
- 关于vue项目中搜索节流的实现
我们经常会遇到这种需求,现在我们在使用百度搜索的时候他们的思想也是根据防抖节流而实现的,至于用防抖还是节流根据自己需求. <template> <input type="t ...