题目:

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

请你返回矩阵中战斗力最弱的 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

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/the-k-weakest-rows-in-a-matrix
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

根据题目中的意思:军人总是排在一行中的靠前位置,1 总是出现在 0 之前,且二维矩阵中不是0就是1。就可以用二分查找,找出一行中最后一个1 的位置,设置位置的初始值pos为-1,则得知最后一个1的位置后,这一行1的个数就为pos+1。

二分查找的细节:

  • 设置left = 0, right = col - 1, pos = -1, mid = left + (right - left + 1) / 2;
  • 循环的条件是:left < right;
  • 如果 mat[i][mid] == 0,则说明mid右边的数都为0,需要移动right到mid-1,搜索范围为[left, mid -1];
  • 如果 mat[i][mid] == 1,则说明mid左边的数都为1,但是mid也有可能是最后一个1的位置,故将 left 移动到 mid,搜索范围为[mid, right](所以mid需要向上取整);
  • 循环结束后:left == right,需要判断left处的值是否为1,为1则pos = left,否则pos还是循环中的pos。

得到每一行的战斗力后,可以将每一行的战斗力和其索引放入到一个小跟堆里面,但是当战斗力相同时,索引较小的更弱,故需要在小根堆中存放战斗力和索引的二元组。

java代码(left < right):

 1 class Solution {
2 public int[] kWeakestRows(int[][] mat, int k) {
3 int row = mat.length, col = mat[0].length;
4 //建立一个小跟堆
5 PriorityQueue<int[]> queue = new PriorityQueue<>((x1, x2) -> {
6 //如果两行个数不相同,就按照个数升序
7 if (x1[0] != x2[0]){
8 return x1[0] - x2[0];
9 } else {
10 //如果两行个数相同,就按照行索引升序
11 return x1[1] - x2[1];
12 }
13 });
14 //二分查找,找到每行1的个数
15 for (int i = 0; i < row; i++){
16 int pos = -1;
17 int left = 0, right = col - 1;
18 while (left < right){
19 int mid = left + (right - left + 1) / 2;
20 if (mat[i][mid] == 0){
21 right = mid - 1;
22 }else {
23 left = mid;
24 //更新pos的位置
25 pos = mid;
26 }
27 }
28 pos = mat[i][left] == 1 ? left : pos;
29 //pos+1:pos为位置,pos+1就为长度
30 queue.offer(new int[]{pos+1, i});
31 }
32 int[] res = new int[k];
33 for (int l = 0; l < k; l++){
34 res[l] = queue.poll()[1];
35 }
36 return res;
37 }
38 }

 python3代码(left <= right):

 1 class Solution:
2 def kWeakestRows(self, mat: List[List[int]], k: int) -> List[int]:
3 row, col = len(mat), len(mat[0])
4 power = list()
5 for i in range(row):
6 left, right, pos = 0, col-1, -1
7 while left <= right:
8 mid = left + (right - left) // 2
9 if mat[i][mid] == 0:
10 right = mid - 1
11 else:
12 left = mid + 1
13 pos = mid
14 power.append((pos+1, i))
15 # 将列表转换为堆
16 heapq.heapify(power)
17 res = list()
18 for i in range(k):
19 res.append(heapq.heappop(power)[1])
20 return res

 小知识:

1.java使用优先队列实现大顶堆和小顶堆,默认是小根堆,当然记不住默认也没有关系

小根堆创建:

 PriorityQueue<Integer> minHeap = new PriorityQueue<>(k,(a,b) -> a-b);

大跟堆创建:

PriorityQueue<Integer> maxHeap = new PriorityQueue<>(k,(a,b) -> b-a);

其中构造器中的k表示创建堆的大小,之后用Lambda表达式快速实现自定义排序。

2.heapq堆的常用方法:

  • heapq.heapify(list): 将列表转换为堆
  • heapq.heappush(heap, itme):heap为定义堆,item增加的元素
  • heapq.heappop(heap):删除并返回最小值,因为堆的特征是heap[0]永远是最小的元素,所以一般都是删除第一个元素
  • heapq.heapreplace(heap.item) :删除并返回最小元素值,添加新的元素值
  • heap.heappushpop(list, itme):判断添加元素值与堆的第一个元素值对比;如果大,则删除并返回第一个元素,然后添加新元素值item;如果小,则返回item,原堆不变
  • heap.nlargest(n, heap):查询堆中的最大n个元素
  • heap.nsmallest(n, heap):查询堆中的最小n个元素

力扣1337(java&python)-矩阵中战斗力最弱的 K 行(简单)的更多相关文章

  1. LeetCode1337矩阵中最弱的K行

    题目 给你一个大小为 m * n 的矩阵 mat,矩阵由若干军人和平民组成,分别用 1 和 0 表示. 请你返回矩阵中战斗力最弱的 k 行的索引,按从最弱到最强排序. 如果第 i 行的军人数量少于第 ...

  2. 力扣(LeetCode)字符串中的单词数 个人题解

    统计字符串中的单词个数,这里的单词指的是连续的不是空格的字符. 请注意,你可以假定字符串里不包括任何不可打印的字符. 示例: 输入: "Hello, my name is John" ...

  3. 力扣(LeetCode)字符串中的第一个唯一字符 个人题解

    给定一个字符串,找到它的第一个不重复的字符,并返回它的索引.如果不存在,则返回 -1. 案例: s = "leetcode" 返回 0. s = "loveleetcod ...

  4. python爬虫中scrapy框架是否安装成功及简单创建

    判断框架是否安装成功,在新建的爬虫文件夹下打开盘符中框输入cmd,在命令中输入scrapy,若显示如下图所示,则说明成功安装爬虫框架: 查看当前版本:在刚刚打开的命令框内输入scrapy versio ...

  5. Python 数据分析中金融数据的来源库和简单操作

    目录 金融数据 pandas-datareader TuShare 金融学图表 案例 金融数据 数据分析离不开数据的获取,这里介绍几种常用的获取金融方面数据的方法. pandas-datareader ...

  6. 力扣 - 剑指 Offer 54. 二叉搜索树的第k大节点

    题目 剑指 Offer 54. 二叉搜索树的第k大节点 思路1 二叉搜索树的特性就是中序遍历结果为递增序列,而题目要求的是第 k 大节点,所以就应该是要遍历结果为降序, 按照先遍历左子树.输出节点.遍 ...

  7. 找到排序矩阵中从小到大第K个数字

    一 题目描述 在一个排序矩阵中找从小到大的第 k 个整数. 排序矩阵的定义为:每一行递增,每一列也递增. 二 题解 由于排序矩阵中的每一行都是递增的,并且每一列都是递增的.从小到大第k个数,实际上就是 ...

  8. 排序矩阵中的从小到大第k个数 · Kth Smallest Number In Sorted Matrix

    [抄题]: 在一个排序矩阵中找从小到大的第 k 个整数. 排序矩阵的定义为:每一行递增,每一列也递增. [思维问题]: 不知道应该怎么加,因为不是一维单调的. [一句话思路]: 周围两个数给x或y挪一 ...

  9. lintcode-401-排序矩阵中的从小到大第k个数

    401-排序矩阵中的从小到大第k个数 在一个排序矩阵中找从小到大的第 k 个整数. 排序矩阵的定义为:每一行递增,每一列也递增. 样例 给出 k = 4 和一个排序矩阵: [ [1 ,5 ,7], [ ...

  10. 在java.ext.dirs中使用环境变量导致crontab执行不成功的问题及解决

    在java.ext.dirs中使用环境变量导致crontab执行不成功的问题及解决 Table of Contents 1. java.ext.dirs的使用和环境变量 2. 问题:在crontab中 ...

随机推荐

  1. C#实现FTP服务端和客户端

    目录 简介 FTP客户端 系统客户端 客户端软件 自定义客户端 FTP服务端 系统服务端 服务端软件 自定义服务端 附件 简介 FTP是FileTransferProtocol(文件传输协议)的英文简 ...

  2. 定义pod的hosts文件(HostAliases)

    通过HostAliases 向 Pod /etc/hosts 文件添加条目 当 DNS 配置以及其它选项不合理的时候,通过向 Pod 的 /etc/hosts 文件中添加条目, 可以在 Pod 级别覆 ...

  3. UE干货| UE虚幻引擎调试神器—控件反射器

    一.打开控件反射器 可以通过窗口→开发者工具→控件反射器 打开: 也可以在umg编辑器上方控件反射器打开. 二.UE控件反射器使用方法 运行项目后,点击控件反射器的"选择可测试命中控件&qu ...

  4. 基础教材系列:Linux原理《趣谈linux》极客时间笔记

    1.电脑一通电,先运行主板上ROM(只读存储器)里写死的程序BIOS,BIOS去找要运行什么操作系统,运行操作系统的第一段代码,创建0号进程,它是这次开机所有进程的爹, 2.然后操作系统代码里先初始化 ...

  5. KingbaseES数据目录结构

    KingbaseES数据库结构 [kingbase@postgres V8]$ tree -LP 2 data/ . ├── data │   ├── base # 存储用户创建的数据库文件及隶属于用 ...

  6. KingbaseES V8R6运维案例之---wal日志解析DDL操作

    ​ 案例说明: 通过sys_waldump解析DDL操作,获取DDL操作的日志条目具体内容. 适用版本: KingbaseES V8R3/R6 一.DDL事务操作对应的wal日志文件 # 查看当前on ...

  7. Java面试题【4】

    28)Java 栈和堆的区别 1 栈:为编译器自动分配和释放,如函数参数.局部变量.临时变量等等 2 堆:为成员分配和释放,由程序员自己申请.自己释放.否则发生内存泄露.典型为使用new申请的堆内容. ...

  8. #结论#CF1776G Another Wine Tasting Event

    题目 给定一个长度为 \(2n-1\) 的字符串,问一组使得 \(n\) 个长度不小于 \(n\) 的区间中字母W的个数相等的字母W的个数 分析 首先结论就是 \(\max_{i=1}^n\{cW[i ...

  9. #期望dp#洛谷 6835 [Cnoi2020]线形生物

    题目 分析 设\(f[i]\)表示由点\(i\)走到点\(i+1\)的期望步数, \(dp[i]\)表示由点1走到点\(i+1\)的期望步数, 那么\(dp\)为\(f\)的前缀和,最后答案为\(dp ...

  10. [原创工具] 病毒整理器V0.98

    最近由于各种需要,所以开发了一个. RT,下载地址: 链接:https://pan.baidu.com/s/1Kxd77-n3fbPZQm_CIH6LTA 提取码:xq8f 很简单的一个小工具,以后还 ...