Largest Submatrix 3 CodeForces - 407D (dp,好题)
大意: 给定矩阵, 求选出一个最大矩形, 满足矩形内每个元素互不相同.
考虑枚举上下左三个边界, 求出最大右边界的位置.
注意到固定上边界, 下边界递推时, 每个左边界对应最大右边界是单调不增的.
所以只需考虑下边界所在行的影响, 与之前的取最小即可.
用$set$求的话复杂度是$O(n^3logn)$, 没有卡过去.
$set$改成$vEB$树的话复杂度可以达到$O(n^3loglogn)$, 或许可以过.
实际上可以发现, 下边界所在行每个点的最大右边界在上边界递减时是非增的, 可以倒序枚举上边界, 这样就可以达到复杂度$O(n^3)$.
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <string.h>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define PER(i,a,n) for(int i=n;i>=a;--i)
using namespace std; const int N = 410;
int n, m, a[N][N], R[N][N];
int l1[N], r1[N], vis[N*N]; int main() {
scanf("%d%d", &n, &m);
REP(i,1,n) REP(j,1,m) scanf("%d",a[i]+j);
memset(R,0x3f,sizeof R);
int ans = 0;
REP(D,1,n) {
REP(i,1,m) r1[i]=m+1,l1[i]=0;
//l1[i] 是行范围在[U,D]内, 列在i左侧, 存在与a[D][i]相等的最接近i的列数
//r1[i] 是行范围在[U,D]内, 列在i右侧, 存在与a[D][i]相等的最接近i的列数
//R[U][i] 为上边界U, 下边界D, 左边界i的矩形的最大右边界
PER(U,1,D) {
int now = 1;
REP(i,1,m) {
now = max(now, i);
while (now<r1[i]&&!vis[a[U][now]]&&!vis[a[D][now]]) {
if (U!=D&&a[U][now]==a[D][now]) break;
vis[a[U][now]] = vis[a[D][now]] = 1;
++now;
}
vis[a[U][i]] = vis[a[D][i]] = 0;
r1[i] = min(r1[i], now);
}
now = m;
PER(i,1,m) {
now = min(now, i);
while (now>l1[i]&&!vis[a[U][now]]&&!vis[a[D][now]]) {
if (U!=D&&a[U][now]==a[D][now]) break;
vis[a[U][now]] = vis[a[D][now]] = 1;
--now;
}
vis[a[U][i]] = vis[a[D][i]] = 0;
l1[i] = max(l1[i], now);
}
REP(i,1,m) {
R[U][i] = min(R[U][i],r1[i]-1);
R[U][l1[i]] = min(R[U][l1[i]],i-1);
}
PER(i,1,m) {
R[U][i] = min(R[U][i], R[U][i+1]);
ans = max(ans, (D-U+1)*(R[U][i]-i+1));
}
}
}
printf("%d\n", ans);
}
还有一种编码非常简单的区间$DP$做法.
设$f_{i,l,r}$为下边界$i$,左右边界$l,r$的最小上边界值, 有转移
$f_{i,l,r}=max\{f_{i-1,l,r},f_{i,l+1,r},f_{i,l,r-1},a_{i,l}$与$a_{1..i,r}$的限制$,a_{1..i,l}$与$a_{i,r}$的限制$\}$
#include <iostream>
#include <algorithm>
#include <cstdio>
#define REP(i,a,n) for(int i=a;i<=n;++i)
using namespace std; const int N = 402;
int n,m,a[N][N],f[N][N],pre[N][N*N]; int main() {
scanf("%d%d", &n, &m);
REP(i,1,n) REP(j,1,m) scanf("%d",a[i]+j);
int ans = 0;
REP(i,1,n) REP(d,1,m) {
for (int l=1,r=d; r<=m; ++l,++r) {
if (l==r) f[l][l]=max(f[l][l],pre[l][a[i][l]]);
else if (a[i][l]==a[i][r]) f[l][r] = i;
else {
f[l][r]=max({f[l][r],f[l][r-1],f[l+1][r],pre[r][a[i][l]],pre[l][a[i][r]]});
}
ans = max(ans, (i-f[l][r])*(r-l+1));
}
REP(j,1,m) pre[j][a[i][j]]=i;
}
printf("%d\n", ans);
}
Largest Submatrix 3 CodeForces - 407D (dp,好题)的更多相关文章
- Array Beauty CodeForces - 1189F (dp,好题)
大意: 定义$n$元素序列$a$的美丽度为 $\min\limits_{1\le i<j\le n}|a_i-a_j|$. 给定序列$a$, 求$a$的所有长为$k$的子序列的美丽度之和. 记 ...
- DP刷题记录
目录 dp刷题记录 codeforces 706C codeforces 940E BZOJ3997 POJ2279 GYM102082B GYM102082D codeforces132C L3-0 ...
- codeforces 407D Largest Submatrix 3
codeforces 407D Largest Submatrix 3 题意 找出最大子矩阵,须满足矩阵内的元素互不相等. 题解 官方做法 http://codeforces.com/blog/ent ...
- Codeforces 148D 一袋老鼠 Bag of mice | 概率DP 水题
除非特别忙,我接下来会尽可能翻译我做的每道CF题的题面! Codeforces 148D 一袋老鼠 Bag of mice | 概率DP 水题 题面 胡小兔和司公子都认为对方是垃圾. 为了决出谁才是垃 ...
- Largest Submatrix(动态规划)
Largest Submatrix Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- POJ-3494 Largest Submatrix of All 1’s (单调栈)
Largest Submatrix of All 1’s Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 8551 Ac ...
- hdu 2870 Largest Submatrix(平面直方图的最大面积 变形)
Problem Description Now here is a matrix with letter 'a','b','c','w','x','y','z' and you can change ...
- [POJ2559&POJ3494] Largest Rectangle in a Histogram&Largest Submatrix of All 1’s 「单调栈」
Largest Rectangle in a Histogram http://poj.org/problem?id=2559 题意:给出若干宽度相同的矩形的高度(条形统计图),求最大子矩形面积 解题 ...
- poj 2955 Brackets (区间dp基础题)
We give the following inductive definition of a “regular brackets” sequence: the empty sequence is a ...
随机推荐
- git前期准备
git小结 设置用户名 git config –global user.name 'itcast' 设置用户名邮箱 git config –global user.email 'itcast' 查看设 ...
- Factor_Analysis
Factor_Analysis(因子分析) Factor Analysis 简书:较好理解的解释,其中公式有一定的推导(仅展现关键步骤,细节大多需要自行补充),基本为结论式. 感性层面理解:首先,明确 ...
- linux下如何让控制台程序后台运行
方法1:./test & 方法2:(./test &) 方法3:nohup 和bg 方法4:screen -mS test /root/test 退出程序需要手动进入screen创 ...
- JVM | JVM体系结构认知
虚拟机 何为虚拟机呢?虚拟机是模拟执行某种指令集体系结构(ISA)的软件,是对操作系统和硬件的一种抽象.其软件模型如下图所示: 计算机系统的这种抽象类似于面向对象编程(OOP)中的针对接口编程泛型(或 ...
- Keil MDK 5代码补全功能设置
这段时间在用Keil5编程,经常会遇到在程序文件头部定义一个全局变量.在后面的编程过程中,经常会要用到这个变量,如果每次再打这个变量名会特别麻烦和浪费时间,我就想着Keil5有没有像vs软件一样的代码 ...
- js刷新页面location.reload()用法
转: js刷新页面location.reload()用法 2018年05月10日 10:23:28 大灰狼的小绵羊哥哥 阅读数 31359更多 分类专栏: [前端面试点滴知识 ] 本文介绍了js刷 ...
- ireport如何拼接sql?
ireport如何拼接sql ireport如何拼接sql? 解决方法: 1.ireport的sql select * from emp as e $P!{whereSQL}; 2.java代码 ...
- java以逗号为分割符拼接字符串的技巧
java以逗号为分割符拼接字符串的技巧 答: 不用那么多if判断,让人思维混乱,直接到最后使用deleteCharAt方法去除最后一个逗号即可. 实现代码如下所示: StringBuffer sb ...
- Django之form表单验证顺序
概述 django框架提供了一个forms类,来处理web开发中的表单相关事项.众所周知,form最常做的是对用户输入的内容进行验证,为此django的forms类提供了全面的内容验证支持. 验证过程 ...
- C++: Unresolved external symbol __imp__fscanf
VS2019中遇到错误error LNK2001: unresolved external symbol __imp__fscanf 解决办法:链接legacy_stdio_definitions.l ...