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)
#include<bits/stdc++.h>
#define IOS ios::sync_with_stdio(false);//不可再使用scanf printf
#define Max(a, b) ((a) > (b) ? (a) : (b))//禁用于函数,会超时
#define Min(a, b) ((a) < (b) ? (a) : (b))
#define Mem(a) memset(a, 0, sizeof(a))
#define Dis(x, y, x1, y1) ((x - x1) * (x - x1) + (y - y1) * (y - y1))
#define MID(l, r) ((l) + ((r) - (l)) / 2)
#define lson ((o)<<1)
#define rson ((o)<<1|1)
#define Accepted 0
#pragma comment(linker, "/STACK:102400000,102400000")//栈外挂
using namespace std;
inline int read()
{
int x=,f=;char ch=getchar();
while (ch<''||ch>''){if (ch=='-') f=-;ch=getchar();}
while (ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
} typedef long long ll;
const int maxn = + ;
const int MOD = ;//const引用更快,宏定义也更快
const int INF = 1e9 + ;
const double eps = 1e-;
int dp1[maxn][];//m = 1 dp1[i][j]表示前i个数中取了j个连续段的最大值
int dp2[maxn][maxn][];//m = 2 dp2[i][j][k]表示第一列前i个数第二列前j个数取了k个连续段的最大值
int s[maxn];
int sum[maxn][];
int main()
{
int n, m, K;
scanf("%d%d%d", &n, &m, &K);
if(m == )
{
for(int i = ; i <= n; i++)scanf("%d", &s[i]), s[i] += s[i - ];//直接记录前缀和
for(int i = ; i <= n; i++)
for(int j = ; j <= K; j++)dp1[i][j] = -INF;//dp[i][0]均为初始化的0
for(int i = ; i <= n; i++)
{
for(int j = ; j <= K; j++)
{
dp1[i][j] = dp1[i - ][j];//不取这个数字
for(int start = ; start <= i; start++)//从start开始一直取到i作为第j段
{
dp1[i][j] = max(dp1[i][j], dp1[start - ][j - ] + s[i] - s[start - ]);
}
}
}
printf("%d\n", dp1[n][K]);
}
else
{
for(int i = ; i <= n; i++)
{
for(int j = ; j <= m; j++)
{
scanf("%d", &sum[i][j]);
sum[i][j] += sum[i - ][j];//记录每一列的前缀和
}
}
for(int i = ; i <= n; i++)for(int j = ; j <= n; j++)for(int k = ; k <= K; k++)dp2[i][j][k] = -INF;
for(int i = ; i <= n; i++)
for(int j = ; j <= n; j++)for(int k = ; k <= K; k++)
{
dp2[i][j][k] = max(dp2[i - ][j][k], dp2[i][j - ][k]);//不取这个数字
for(int start = ; start <= i; start++)//第一列从start开始一直取到i作为第k个矩阵
dp2[i][j][k] = max(dp2[i][j][k], dp2[start - ][j][k - ] + sum[i][] - sum[start - ][]);
for(int start = ; start <= j; start++)//第二列从start开始一直取到j作为第k个矩阵
dp2[i][j][k] = max(dp2[i][j][k], dp2[i][start - ][k - ] + sum[j][] - sum[start - ][]);
if(i == j)//两列从start开始一直取到i作为第k个矩阵
{
for(int start = ; start <= i; start++)
dp2[i][j][k] = max(dp2[i][j][k], dp2[start - ][start - ][k - ] + sum[i][] + sum[i][] - sum[start - ][] - sum[start - ][]);
}
}
printf("%d\n", dp2[n][n][K]);
}
return Accepted;
}
BZOJ 1084 最大子矩阵 dp的更多相关文章
- BZOJ 1084 最大子矩阵
http://www.lydsy.com/JudgeOnline/problem.php?id=1084 思路:分m=1和m=2操作 #include<algorithm> #includ ...
- [SCOI2005][BZOJ 1084]最大子矩阵
Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第一行为n,m,k(1≤n≤100,1≤m≤2 ...
- [BZOJ 1084] [SCOI2005] 最大子矩阵 【DP】
题目链接:BZOJ - 1084 题目分析 我看的是神犇BLADEVIL的题解. 1)对于 m = 1 的情况, 首先可能不取 Map[i][1],先 f[i][k] = f[i - 1][k]; ...
- 【SCOI2005】 最大子矩阵 BZOJ 1084
Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第一行为n,m,k(1≤n≤100,1≤m≤2 ...
- BZOJ 1084: [SCOI2005]最大子矩阵 DP
1084: [SCOI2005]最大子矩阵 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=1084 Description 这里有一个n* ...
- 【BZOJ 1084】 1084: [SCOI2005]最大子矩阵 (DP)
1084: [SCOI2005]最大子矩阵 Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. Input 第 ...
- 【BZOJ 1084】 [SCOI2005]最大子矩阵(DP)
题链 http://www.lydsy.com/JudgeOnline/problem.php?id=1084 Description 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩 ...
- bzoj 1084: [SCOI2005]最大子矩阵【dp】
分情况讨论,m=1的时候比较简单,设f[i][j]为到i选了j个矩形,前缀和转移一下就行了 m=2,设f[i][j][k]为1行前i个,2行前j个,一共选了k个,i!=j的时候各自转移同m=1,否则转 ...
- BZOJ 1084 (SCOI 2005) 最大子矩阵
1084: [SCOI2005]最大子矩阵 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 3560 Solved: 1779 [Submit][Sta ...
随机推荐
- js的一些妙用
在一个数组上 直接附加上另一个数组: Array.prototype.push.apply(array1, array2); 将对象转换成一个数组: Array.prototype.slice.ca ...
- [转]Grunt 新手一日入门
本文转自:http://yujiangshui.com/grunt-basic-tutorial/ 当时学习 Grunt 的时候,真是很头疼.分了两个时间段,学习了两次才硬啃下来,之后才能用在项目中. ...
- Varint数值压缩算法
Varint 是一种紧凑的表示数字的方法.它用一个或多个字节来表示一个数字,值越小的数字使用越少的字节数.这能减少用来表示数字的字节数.比如对于 int32 类型的数字,一般需要 4 个 byte 来 ...
- SQL 事务与锁
了解事务和锁 事务:保持逻辑数据一致性与可恢复性,必不可少的利器. 锁:多用户访问同一数据库资源时,对访问的先后次序权限管理的一种机制,没有他事务或许将会一塌糊涂,不能保证数据的安全正确读写. 死锁: ...
- golang学习之defer
golang中的defer通常用于执行一些资源释放性操作,比如open/close.connect/disconnect.lock/unlock等,对defer理解主要记住以下三点: 1.defer ...
- WCF4.0安装 NET.TCP启用及常见问题
WCF4.0安装及NET.TCP启用 WCF 4.0 一般默认安装.net Framework 4.0的时候已经安装. 但如果先装.net framework 4.0,后装IIS,就会出现问题.需要重 ...
- spss C# 二次开发 学习笔记(六)——Spss统计结果的输出
Spss的二次开发可以很简单,实例化一个对象,然后启用服务,接着提交命令,最后停止服务. 其中重点为提交命令,针对各种统计功能需求,以及被统计分析的数据内容等,命令的内容可以很复杂,但也可以简单的为一 ...
- PAT 1074. Reversing Linked List
#include <cstdio> #include <cstdlib> #include <iostream> #include <unordered_ma ...
- jQuery bind()与delegate()的区别
笔试题: bind()与delegate()的区别主要有三点: 1 绑定目标 .bind直接绑在目标元素上 .delegate绑在父元素上 2 监听个数 .bind监听个数多——每个目标元素都需要添 ...
- mysql 报错 ‘u'Subquery returns more than 1 row'’
watch_course_sql ) , '%%Y-%%m-%%d %%T') regtime, a.username FROM bskuser a where a.UserName in (sele ...