程序员代码面试指南第二版 9.求最大子矩阵的大小(要掌握单调栈结构)
题目描述(不含重复值的数组)
题目描述
给定一个整型矩阵 map,其中的值只有 0 和 1 两种,求其中全是 1 的所有矩形区域中,最大的矩形区域里 1 的数量。
输入描述:
第一行输入两个整数 n 和 m,代表 n*m 的矩阵
接下来输入一个 n*m 的矩阵
输出描述:
输出其中全是 1 的所有矩形区域中,最大的矩形区域里 1 的数量。
示例1
输入
1 4
1 1 1 0
输出
3
说明
最大的矩形区域有3个1,所以返回3
第一次做; 时间复杂度O(M*N); 每一行都走一遍单调栈结构(带重复值,但是重复值仍然按照小于的情况处理), 分为两个阶段: 遍历阶段; 清算阶段; 没有左边, 让左边为-1, 这样左边的下一个位置是0, 不影响结果; 没有右边, 让右边为arr.length, 这样右边的前一个位置是arr.length-1; 需要进行单调栈处理的数组height的每个元素的含义是: 以当前位置为底, 往上有多少个连续的1, 包括当前位置, height[j] = matrix[i][j]==1 ? height[j]+1 : 0, 有点动态规划转移方程的感觉
import java.util.Scanner;
import java.util.Stack;
public class Main{
public static void main(String[] args){
Scanner sc = new Scanner(System.in);
String curr = sc.nextLine();
int rows = Integer.valueOf(curr.split(" ")[0]);
int cols = Integer.valueOf(curr.split(" ")[1]);
int[][] matrix = new int[rows][cols];
for(int i=0; i<rows; i++){
for(int j=0; j<cols; j++){
matrix[i][j] = Integer.valueOf(sc.nextInt());
}
}
int res = MaxRecArea(matrix);
System.out.println(res);
}
public static int MaxRecArea(int[][] matrix){
if(matrix==null || matrix.length==0 || matrix[0].length==0)
return 0;
int rows = matrix.length;
int cols = matrix[0].length;
int[] height = new int[cols];
int max = 0;
for(int i=0; i<rows; i++){
for(int j=0; j<cols; j++){
height[j] = matrix[i][j] == 1 ? height[j] + 1 : 0;
}
max = Math.max(max, monotonicStack(height));
}
return max;
}
public static int monotonicStack(int[] arr){
Stack<Integer> s = new Stack<>();
int max = 0;
for(int i=0; i<arr.length; i++){
if(s.isEmpty())
s.push(i);
else{
if(arr[i] > arr[s.peek()])
s.push(i);
else{
while(!s.isEmpty() && arr[i] <= arr[s.peek()]){
int curr = s.pop();
int newTop = s.isEmpty() ? -1 : s.peek();
max = Math.max(max, (i - 1 - newTop)*arr[curr]);
}
s.push(i);
}
}
}
while(!s.isEmpty()){
int curr = s.pop();
int newTop = s.isEmpty() ? -1 : s.peek();
max = Math.max(max, (arr.length - 1 - newTop)*arr[curr]);
}
return max;
}
}