Largest Submatrix
给出一个\(n\times m\)的网格,网格里只放有字符a,b,c,d,w,x,,z,现在你可以将其中的w换成a,b,把x换成b,c,把y换成a,c,把z换成a,b,c,询问换完以后最大的子矩阵大小,使其包含一样的字符,\(1 ≤ m, n ≤ 1000\)。
解
注意到字符很少,我们可以钦定最大的子矩阵的字符,也就是尽量把字符变成变成所钦定,正确性显然。
于是接下来问题就变成了如何快速找到这个矩阵了,接下来简要提一下做法,如果初学者看不懂可以看一下这道题目Largest Rectangle in a Histogram,注意到这个类似与最大相同子矩阵的模型,我们只要按行枚举,再枚举列,暴力预处理每一列在以该行为起点上向上所能最长的连续所钦定的字符,预处理出这个问题转化为有m个宽度为1矩形,第i个矩形高度为\(h_i\),从左至右下端对齐x轴,求其中最大的子矩形。
对于这个问题,按照枚举的思想,先枚举第几个矩形i,以这个矩形为高向左右延伸能最大的矩形,也就是碰到第一个矩形高度小于i的位置,这是单调性问题,考虑单调队列维护,维护一个矩形高度单调递增的单调队列。
于是队列中保存两个值,一个是该矩形向左最远能延伸位置,一个是该矩形的高度,将要入队元素如果能满足单调性,则不予考虑,否则弹出队尾,而队尾元素未被弹出必然是从队尾所对应的位置到当前要入队的位置高度都要比队尾高度大,这也是单调队列的性质,这样队尾所对应矩形最远延伸位置即已经保存下来的最左边位置和入队元素对应位置的上一个,而至于新入队的矩形所对应的最左延升位置即队尾的最左延伸位置,依次维护下去就可以了。
时间复杂度不难得知\(O(nm)\)。
参考代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#define il inline
#define ri register
#define Size 1500
using namespace std;
char A[Size][Size];
bool B[Size][Size];
int n,m,T[Size],R,h[Size],l[Size],ans;
il void work();
template<class free>
il free Max(free,free);
int main(){
while(scanf("%d%d",&n,&m)!=EOF){ans=0;
for(int i(1);i<=n;++i)
scanf("%s",A[i]+1);
for(int i(1),j;i<=n;++i)
for(j=1;j<=m;++j)
if(A[i][j]=='a'||A[i][j]=='w'||A[i][j]=='y'||A[i][j]=='z')B[i][j]|=true;
work(),memset(B,0,sizeof(B));
for(int i(1),j;i<=n;++i)
for(j=1;j<=m;++j)
if(A[i][j]=='b'||A[i][j]=='w'||A[i][j]=='x'||A[i][j]=='z')B[i][j]|=true;
work(),memset(B,0,sizeof(B));
for(int i(1),j;i<=n;++i)
for(j=1;j<=m;++j)
if(A[i][j]=='c'||A[i][j]=='x'||A[i][j]=='y'||A[i][j]=='z')B[i][j]|=true;
work(),memset(B,0,sizeof(B)),printf("%d\n",ans);
}
return 0;
}
il void work(){
for(int i(1),j;i<=n;++i){
for(j=1;j<=m;++j){
h[j]=0,l[j]=j;while(B[i+h[j]][j])++h[j];
while(0<R&&h[T[R]]>h[j])
ans=Max((j-l[T[R]])*h[T[R]],ans),l[j]=l[T[R]],--R;
T[++R]=j;
}while(0<R)ans=Max((j-l[T[R]])*h[T[R]],ans),--R;
}
}
template<class free>
il free Max(free a,free b){
return a>b?a:b;
}
Largest Submatrix的更多相关文章
- 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 ...
- Largest Submatrix of All 1’s
Given a m-by-n (0,1)-matrix, of all its submatrices of all 1’s which is the largest? By largest we m ...
- codeforces 407D Largest Submatrix 3
codeforces 407D Largest Submatrix 3 题意 找出最大子矩阵,须满足矩阵内的元素互不相等. 题解 官方做法 http://codeforces.com/blog/ent ...
- Largest Submatrix of All 1’s(思维+单调栈)
Given a m-by-n (0,1)-matrix, of all its submatrices of all 1's which is the largest? By largest we m ...
- POJ 3494 Largest Submatrix of All 1’s 单调队列||单调栈
POJ 3494 Largest Submatrix of All 1’s Description Given a m-by-n (0,1)-matrix, of all its submatrice ...
- POJ - 3494 Largest Submatrix of All 1’s 单调栈求最大子矩阵
Largest Submatrix of All 1’s Given a m-by-n (0,1)-matrix, of all its submatrices of all 1’s which is ...
- HDU 2870 Largest Submatrix (单调栈)
http://acm.hdu.edu.cn/showproblem.php? pid=2870 Largest Submatrix Time Limit: 2000/1000 MS (Java/Oth ...
- MINSUB - Largest Submatrix
MINSUB - Largest Submatrix no tags You are given an matrix M (consisting of nonnegative integers) a ...
随机推荐
- eduCF#61 F. Clear the String /// 区间DP 消除连续一段相同字符 全部消完的最少次数
题目大意: 给定字符串 每次消除可消除连续的一段相同的字符的子串 求消除整个字符串的最少消除次数 #include <bits/stdc++.h> using namespace std; ...
- 微信小程序之评分页面
首先给大家看看做好的效果图: 一.接下来我们说一下评分这个功能: 实际上就是一个简单的js,首先我们遍历出小星星,此时默认给的五星好评,在给他们一个点击事件,当点击时,我们获取到当前点击的是第几颗:代 ...
- window 对象常见的事件
1.页面加载事件 方式1:window.onload = function(){ } window.addEventListener('load',function(){ }) window.onlo ...
- Flask理论基础(一)加载配置文件
一.修改/新增配置项 1.使用配置文件 app.config.from_pyfile("config.cfg") 如上 config.cfg 可以是任意后缀的文本文件,需要与app ...
- Echart中X轴数据过多时横向拉动展示
chart.setOption( { tooltip: { trigger: 'axis' }, toolbox: { feature: { saveAsImage: {} } }, grid: { ...
- eclipse背景设置什么颜色缓解眼睛疲劳
eclipse背景设置什么颜色缓解眼睛疲劳 1.打开window->Preference,弹出Preference面板 2.展开General标签,选中Editors选项,展开. 3.选中 Te ...
- Java中JNI的使用详解第五篇:C/C++中操作Java中的数组
在Java中数组分为两种: 1.基本类型数组 2.对象类型(Object[])的数组(数组中存放的是指向Java对象中的引用) 一个能通用于两种不同类型数组的函数: GetArrayLength(ja ...
- NX二次开发-UFUN获取球的参数UF_MODL_ask_sphere_parms
NX11+VS2013 #include <uf.h> #include <uf_modl.h> #include <uf_ui.h> UF_initialize( ...
- 牛客多校第八场 G Gemstones 栈/贪心
题意: 对于一个序列,把可以把连着三个相同的字母拿走,问最多拿走多少组. 题解: 直接模拟栈,三个栈顶元素相同则答案+1,并弹出栈 #include<bits/stdc++.h> usin ...
- JS中 reduce() 的用法
过去有很长一段时间,我一直很难理解 reduce() 这个方法的具体用法,平时也很少用到它.事实上,如果你能真正了解它的话,其实在很多地方我们都可以用得上,那么今天我们就来简单聊聊JS中 reduce ...