最新【每日挠头算法题】LeetCode 1337,身为一个大数据开发程序员Context都没弄明白

img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化资料的朋友,可以戳这里获取

文章目录

如果无聊的话,就来逛逛 我的博客栈 吧! 🌹

一、题目描述

今天的题目其实可以暴力求解,但是我们今天主要为了讲解 二分 和 堆,以练习为主~

链接1337. 矩阵中战斗力最弱的 K 行

描述

给你一个大小为 m * n 的矩阵 mat,矩阵由若干军人和平民组成,分别用 10 表示。

请你返回矩阵中战斗力最弱的 k 行的索引,按从最弱到最强排序。

如果第 i 行的军人数量少于第 j 行,或者两行军人数量相同但 i 小于 j,那么我们认为第 i 行的战斗力比第 j 行弱。

军人 总是 排在一行中的靠前位置,也就是说 1 总是出现在 0 之前。

示例1

输入:mat = 
[[1,1,0,0,0],
 [1,1,1,1,0],
 [1,0,0,0,0],
 [1,1,0,0,0],
 [1,1,1,1,1]], 
k = 3
输出:[2,0,3]
解释:
每行中的军人数目:
行 0 -> 2 
行 1 -> 4 
行 2 -> 1 
行 3 -> 2 
行 4 -> 5 
从最弱到最强对这些行排序后得到 [2,0,3,1,4]

示例2

输入:mat = 
[[1,0,0,0],
 [1,1,1,1],
 [1,0,0,0],
 [1,0,0,0]], 
k = 2
输出:[0,2]
解释: 
每行中的军人数目:
行 0 -> 1 
行 1 -> 4 
行 2 -> 1 
行 3 -> 1 
从最弱到最强对这些行排序后得到 [0,2,3,1]

提示

  • m == mat.length
  • n == mat[i].length
  • 2 <= n, m <= 100
  • 1 <= k <= m
  • matrix[i][j] 不是 0 就是 1

二、思路及代码实现

首先梳理一下题目大意:

给定一个矩阵,矩阵元素由 10 组成,1 为军人,0 为平民。军人数量就是矩阵的战斗力。军人 1 出现在矩阵每一行的 靠前位置

如果 第 i 行 1 数量少于第 j 行,或者第 i 行和第 j 行 1 的数量相同,但是 i < j 那么认为 第 i 行的战斗力 比第 j 行弱

题目要求返回前 k 行的索引,就是按照顺序返回 1 最少的前 k 行。

所以这道题目先得求出每行的 1 的个数

求每行 1 的个数可以通过遍历每一行来实现,但是我认为最好的方法还是 二分

由于二维数组每行是从 1 开始,到 0 结束,所以数组整体是有序的。那么我只需要二分出 1右边界点 就可以了。

但是要求出前 k 行战斗力最弱的索引仅有 战斗力 是没用的,我们需要之后比较战斗力的同时返回相应索引,并且对于战斗力相同的情况下需要比较 索引的大小 。所以需要考虑一下 用什么存储数据

了解了这些,我们接下来讲解我们的主要解法。解法分为两种:二分 + 排序二分 + 小堆

1. 二分 + 排序

这里我们采用的二分方式是 二分出右边界点 ,用之前的二分模板:

// 区间[l, r]被划分成[l, mid - 1]和[mid, r]时使用:
int bsearch\_2(int l, int r)
{
    while (l < r)
    {
        int mid = l + r + 1 >> 1;
        if (check(mid)) l = mid;
        else r = mid - 1;
    }
    return l;
}

每次二分需要将 每行中1的个数当前行数 存储到对应的空间中。

所以我们可以定义一个结构体,用来专门存放两种类型的数据:

typedef struct data
{
    int combat; // 战斗力
    int row; // 行数
}data;

紧接着动态开辟一个结构体数组 tmp ,用来存储数据;一个 k 个大小的数组 res 作为返回的数组。

在二分的过程中

  • 如果二分出来的边界点的值 不等于 1 ,说明二分结果错误,那么此行战斗力 combat 为 0 ,存到正确位置
  • 如果二分出来的边界点的值 等于 1 ,说明二分结果正确,将 l(r) + 1 存入结构体数组的对应位置

有了结构体数组,那么进行排序就好了,这里直接使用 qsort ,注意需要处理一下 特殊情况i 行和第 j 行 1 的数量相同,但是 i < j 那么认为 第 i 行的战斗力 比第 j 行弱

最后将数据存入返回数组中,返回即可。

过程相对简单,直接上代码:

typedef struct data
{
    int combat; // 战斗力
    int row; // 行
}data;

int cmp(const void\* e1, const void\* e2)
{
    data\* ee1 = (data\*)e1;
    data\* ee2 = (data\*)e2;
    // 战斗力大小 或 战斗力相等 行数不同
    return (ee1->combat > ee2->combat) || (ee1->combat == ee2->combat && ee1->row > ee2->row);
}

int\* kWeakestRows(int\*\* mat, int matSize, int\* matColSize, int k, int\* returnSize)
{
    // 答案数组
    int\* res = (int\*)malloc(sizeof(int) \* k);
    data\* tmp = (data\*)malloc(sizeof(data) \* matSize);
    int col = \*matColSize;
    \*returnSize = k;

    // 二分,将数据存入 tmp 中
    for (int i = 0; i < matSize; i++) {
        int l = 0, r = col - 1;
        while (l < r) {
            int mid = l + r + 1 >> 1;
            if (mat[i][mid] == 1) {
                l = mid;
            } else if (mat[i][mid] == 0) {
                r = mid - 1;
            }
        }
        if (mat[i][l] != 1) {
            tmp[i].combat = 0; // 无战斗力
        } else {
            tmp[i].combat = l + 1;
        }
        tmp[i].row = i; // 存入索引
    }
    
    qsort(tmp, matSize, sizeof(data), cmp);

    for (int i = 0; i < k; i++)
    {
        res[i] = tmp[i].row;
    }

    return res;
}

image-20221218144735841

2. 二分 + 堆

这种方法其实是我第一次就想到的,但是中途调试了很久,觉得这种思路比排序难一些就把它放到第二个。

先讲常规操作:

这里我们采用的存储结构是用 二维数组 ,将其定义为全局变量 int heap[100][2]

将每一行看作是一个元素,存放 战斗力索引

img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化资料的朋友,可以戳这里获取

转存中…(img-4KZthsmJ-1715817203952)]
[外链图片转存中…(img-dU3k3ZjE-1715817203952)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化资料的朋友,可以戳这里获取

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值