leetcode 1727. 重新排列后的最大子矩阵 中等

给你一个二进制矩阵 matrix ,它的大小为 m x n ,你可以将 matrix 中的 按任意顺序重新排列。

请你返回最优方案下将 matrix 重新排列后,全是 1 的最大子矩阵面积。

示例 1:

复制代码
输入:matrix = [[0,0,1],[1,1,1],[1,0,1]]
输出:4
解释:你可以按照上图方式重新排列矩阵的每一列。
最大的全 1 子矩阵是上图中加粗的部分,面积为 4 。

示例 2:

复制代码
输入:matrix = [[1,0,1,0,1]]
输出:3
解释:你可以按照上图方式重新排列矩阵的每一列。
最大的全 1 子矩阵是上图中加粗的部分,面积为 3 。

示例 3:

复制代码
输入:matrix = [[1,1,0],[1,0,1]]
输出:2
解释:由于你只能整列整列重新排布,所以没有比面积为 2 更大的全 1 子矩形。

示例 4:

复制代码
输入:matrix = [[0,0],[0,0]]
输出:0
解释:由于矩阵中没有 1 ,没有任何全 1 的子矩阵,所以面积为 0 。

提示:

  • m == matrix.length
  • n == matrix[i].length
  • 1 <= m * n <= 10^5
  • matrix[i][j] 要么是 0 ,要么是 1

分析:枚举子矩形的底边(最后一行),定义 heights[j] 表示从 matrix[i][j] 往上有多少个连续的 1(柱子的高度),问题变成:可以重排 heights。重排后,对于 heights 的连续子数组,子数组长度(矩形底边长)× 子数组最小值(矩形的高),即为全 1 子矩形的面积。枚举找到面积最大的子矩形。枚举子数组的长度 k=1,2,...,n,贪心地把 heights 最大的 k 个数排在一起,就可以让子数组的最小值(矩形的高)尽量大,从而得到最大的矩形面积。

对于 heights 的计算,如果 matrix[i][j]=0,那么 heights[j]=0。否则,把高度增加 1。形象地说,就是在柱子下面垫一块石头,把柱子抬高。

cpp 复制代码
class Solution {
public:
    int largestSubmatrix(vector<vector<int>>& matrix) {
        int m=matrix.size(),n=matrix[0].size(),ans=0;
        // printf("m=%d n=%d\n",m,n);
        vector<int>height;
        for(int i=0;i<m;++i)
        {
            // printf("i=%d\n",i);
            for(int j=0;j<n;++j)
            {
                if(!i)height.push_back(matrix[i][j]);
                else if(matrix[i][j])height[j]+=matrix[i][j];
                else height[j]=0;
            }
            vector<int>temp=height;
            sort(temp.begin(),temp.end());
            int len=0x3fffffff;
            for(int j=temp.size()-1,t=1;j>=0&&temp[j]>0;--j,++t)
            {
                len=min(len,temp[j]);
                ans=max(ans,len*t);
                // printf("i=%d j=%d t=%d len=%d temp=%d ans=%d\n",i,j,t,len,temp[j],ans);
            }
        }
        return ans;
    }
};
相关推荐
倦王8 小时前
力扣日刷47-补
python·算法·leetcode
无限进步_9 小时前
【C++】巧用静态变量与构造函数:一种非常规的求和实现
开发语言·c++·git·算法·leetcode·github·visual studio
凌波粒10 小时前
LeetCode--344.反转字符串(字符串/双指针法)
算法·leetcode·职场和发展
啊哦呃咦唔鱼10 小时前
LeetCode hot100-543 二叉树的直径
算法·leetcode·职场和发展
样例过了就是过了11 小时前
LeetCode热题100 爬楼梯
c++·算法·leetcode·动态规划
ThisIsMirror11 小时前
leetcode 452 Arrays.sort()排序整数溢出、Integer.compare(a[1], b[1])成功的问题
算法·leetcode
x_xbx13 小时前
LeetCode:438. 找到字符串中所有字母异位词
算法·leetcode·职场和发展
派大星~课堂13 小时前
【力扣-94.二叉树的中序遍历】Python笔记
笔记·python·leetcode
AI成长日志14 小时前
【笔面试算法学习专栏】堆与优先队列实战:力扣hot100之215.数组中的第K个最大元素、347.前K个高频元素
学习·算法·leetcode
6Hzlia14 小时前
【Hot 100 刷题计划】 LeetCode 45. 跳跃游戏 II | C++ 贪心算法最优解题解
c++·leetcode·游戏