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;
    }
};
相关推荐
脱氧核糖核酸__19 小时前
LeetCode热题100——234.回文链表(两种解法)
c++·算法·leetcode·链表
生信研究猿20 小时前
leetcode 416. 分割等和子集
算法·leetcode·职场和发展
6Hzlia21 小时前
【Hot 100 刷题计划】 LeetCode 24. 两两交换链表中的节点 | C++ 精准指针舞步
c++·leetcode·链表
踩坑记录1 天前
leetcode 92. 反转链表 II 区间反转(不是整条链表反转)
leetcode·链表
6Hzlia1 天前
【Hot 100 刷题计划】 LeetCode 148. 排序链表 | C++ 归并排序自顶向下
c++·leetcode·链表
im_AMBER1 天前
Leetcode 162 除了自身以外数组的乘积 | 接雨水
开发语言·javascript·数据结构·算法·leetcode
啊哦呃咦唔鱼1 天前
leetcodehot100-347. 前 K 个高频元素
数据结构·算法·leetcode
玛丽莲茼蒿1 天前
Leetcode hot100 多数元素【简单】
算法·leetcode·职场和发展
AbandonForce1 天前
Map类:pair键值对|map的基本操作|operator[]
开发语言·c++·算法·leetcode
田梓燊1 天前
力扣:146.LRU 缓存
算法·leetcode·缓存