数组中最大子矩阵,最简便的解法

题目初试
遇到一个好人,可以改变一生;遇到一本好书,又何尝不是呢?

最近在翻阅 左程云先生的《程序员代码面试指南–IT名企算法与数据结构题目最优解》时就非常的有感悟。建议有这方面爱好的博友,也去观摩观摩。

书中讲解的基于栈的数组的最大矩阵的算法很经典,但是博主能力有限,没能彻底的领悟该算法的精髓,但是根据这个思想,博主想出了一种简易的应对该类问题的算法,现概述如下。


核心思想

先来看一张图吧,我们就可以大致的理解了。

算法思想

如图,每一个轮次都是一次运算,而我们的核心就是针对这每一个轮次的内部的运算。

计算出每一层连续不间断的最大长度

也就是说,我们是最这个数组进行由下至上的轮次计算,然后针对每一轮就可以计算出本轮次可以得出的连续最大子矩阵的面积。然后只需要比较每一个轮次的最大的那个数据,返回就可以求出该数组最大的连续的子矩阵的面积了。

代码

好了,有了上面的核心思想的铺垫,我们就可以着手编写代码了。(虽然我也觉得自己并没有说的很清楚,见谅见谅)。

/**
 * @Date 2016年9月9日
 *
 * @author 郭  璞
 *
 */
package stack_and_queue;

/**
 * @author 郭 璞<br>
 *         根据数组来计算连续的最大的矩形区域的面积
 */
public class MaxRectangle {

    public static void main(String[] args) {
        Integer[] arr = { 2, 1, 3, 5, 7, 6, 4 };
        Integer maxRectangle = maxRectangleArea(arr);
        System.out.println("数组中最大的连续的矩形区域的面积为: " + maxRectangle);
    }

    /**
     * @param arr
     * @return 数组中连续矩形区域的最大面积
     */
    private static Integer maxRectangleArea(Integer[] arr) {
        int[] result = new int[arr.length];
        // 对数组进行遍历式的计算,由底向上不间断的连续长度
        for (int i = 1; i <= arr.length; i++) {
            // 当前轮次 中实现对连续长度的累加的临时取值
            int templen = 0;
            // 记录本轮高度下的最大连续长度
            int templen_max = 0;
            // 内层循环应该是从数组首部开始,而从当先层下标开始会导致前面部分数据的丢失
            for (int j = 1; j <= arr.length; j++) {
                if (arr[j - 1] >= i) {
                    templen += 1;
                    templen_max = templen;
                } else {
                    templen = 0;
                }
            }
            result[i - 1] = i * templen_max;
            // System.out.println("第" + i + "层连续不间断的最大长度为:" + templen_max);
        }

        // 求得结果集数组中数值最大的那个数,即为所求连续区域中的最大的矩形域的面积
        int maxArea = 0;
        for (int i = 0; i < result.length; i++) {
            maxArea = maxArea > result[i] ? maxArea : result[i];
        }

        // 将所求得的最大连续矩形域的面积返回
        return maxArea;
    }

}

代码中的注释也比较的全面,就不再过多的赘述了。

测试

下面就对数组进行测试。首先以本文开头图片上所示的数组来进行测试吧。


Integer[] arr = {2,1,3,5,7,6,4}
···
数组中最大的连续的矩形区域的面积为: 16

然后我们修改一下数组中元素的值,来进一步的测试看看结果是否正确。
改变数组进一步测试

Integer[] arr = {2,1,3,1,7,6,4}
···

数组中最大的连续的矩形区域的面积为: 12

经博主本人亲自测试,该算法可以正常的工作。 :)

优化部分

说道优化部分,首先我们可以看出的估计便是最后的那步求结果集数组中的最大值了吧。

确实是的,我们其实可以另外申请一个变量,来记录到目前为止的轮次的最大的那个子矩阵的面积的。不过 这点优化而言起到的作用不是很大,时间复杂度也没有什么比较大的改善。

另外一点,我觉的可以比较好的切入点就是对每一个轮次的进行运算的时候添加一个判断,来决定当前轮次之前是否往下循环。如果数组中的元素的波动比较大的话,优化的程度还是很不错的。

总结

这个小算法比较精巧,唯一比较缺憾的地方就是时间复杂度稍微的有点大了。如果读者在寻求一种时间复杂度比较小的算法的话,请绕道咯。

如果只是想寻求一个结果,还是不错的。至少比暴力方式的计算效率高多了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值