程序员代码面试指南第二版 9.求最大子矩阵的大小(要掌握单调栈结构)

welcome to my blog

程序员代码面试指南第二版 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;
        //每一个元素含义, 以当前索引为底, 向上有多少个连续的1, 包括当前位置
        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();
                        /*
                        矩形面积的计算公式:
                        curr向左可以到达s.peek()+1
                        curr向右可以到达i-1
                        所以矩形面积为: (i-1-(s.peek()+1)+1)*height[i] = (i - 1 - s.peek())*height[i]
                        */
                        //没有左边的时候, 把左边当作-1, 那么左边的下一个位置就是0, 不影响最终结果(这里要想清楚)
                        int newTop = s.isEmpty() ? -1 : s.peek();
                        max = Math.max(max, (i - 1 - newTop)*arr[curr]);
                    }
                    s.push(i);
                }
            }
        }
        
        //清算阶段; 栈顶索引没有右边了, 把右边当作arr.length,  arr.length的前一个位置就是arr.length-1, 不影响最终结果(这里要想清楚)
        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;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值