一天一大 leet(有序矩阵中第 K 小的元素)难度:中等-Day20200702

在这里插入图片描述

题目:有序矩阵中第 K 小的元素

给定一个 n x n 矩阵,其中每行和每列元素均按升序排序,找到矩阵中第 k 小的元素。
请注意,它是排序后的第 k 小元素,而不是第 k 个不同的元素。

示例
matrix = [
   [ 1,  5,  9],
   [10, 11, 13],
   [12, 13, 15]
],
k = 8,
返回 13。
提示

你可以假设 k 的值永远是有效的,1 ≤ k ≤ n2 。

抛砖引玉
暴力排序
  • 首先想到的是先拼接数组
  • 后排序直接取第 k-1 位(索引 0,第一小)

img

/**
 * @param {number[][]} matrix
 * @param {number} k
 * @return {number}
 */
var kthSmallest = function (matrix, k) {
  let sorted = []
  for (let i = 0; i < matrix.length; i++) {
    sorted = [...sorted,...matrix[i]]
  }
  sorted.sort((a, b) => a - b)
  return sorted[k - 1]
}

题目中每行和每列元素均按升序排序这个应该可以作为优化的点

1234
11121314
21222324
31323334

随便找一个符合规则的matrix,找下规则(row表示行,i表示行索引,column,表示列j表示列索引)

  • matrix[0][0]最小,matrix[row-1][column-1],最大
  • 那么当指针在matrix[i][j],下一个比他大的数会出现的位置会在matrix[x][y]到matrix[row-1][column-1]
    • x范围:i到row-1
    • y范围:当x为i时(j到column-1),当x为i++(大于i)时,在0到j之间可能也会有下一个比他大的数

想要单次遍历逐个递增的来统计第k小的数,会发现下一个比他大的数的区值范围在一个梯形范围内很难具体定位,
换个思路,既然指定一个数,我可以定位到大于他的范围,那假设我已经知道了第k小的元素是m那么,直接统计小于他的数是不是k-1个就可以验证m的真实性了。


二分法
  • matrix[0][0]到matrix[row-1][column-1]中任意取一个数mid做第k小的数,(取中间值,会最快取到想要的值)
  • 遍历matrix检查小于mid的数是否等于k
    • 大于k,则说明m取大了,那么再从matrix[0][0]到m中取个中间值
    • 小于k,则说明m取小了,那么再从m到matrix[row-1][column-1]中取个中间值
    • 等于k,理论上是取到看第k个,但是因为取的是中间值,也许这个数并不在matrix,那么只能再进行范围划分知道上下线重合
var kthSmallest = function (matrix, k) {
    let n = matrix.length;
    let left = matrix[0][0];
    let right = matrix[n - 1][n - 1];
    while (left < right) {
        let mid = left + parseInt((right - left) / 2, 10);
        if (check(matrix, mid, k, n)) {
            right = mid;
        } else {
            left = mid + 1;
        }
    }
    return left;
    
    function check(matrix, mid, k, n) {
        let i = n - 1;
        let j = 0;
        let num = 0;
        while (i >= 0 && j < n) {
            if (matrix[i][j] <= mid) {
                num += i + 1;
                j++;
            } else {
                i--;
            }
        }
        return num >= k;
    }
}
其他解法
  • 一行一行合并
  • 之后合并的行循环按顺序插入到上一次合并的数组中
  • 利用reduce第一个参数做合并的目标数组,异常逐行合并到其中

reduce方法接收一个函数作为累加器,数组中的每个值(从左到右)开始缩减,最终计算为一个值

var kthSmallest = function(matrix, k) {
  if(matrix.length < 1) return 0
  let arr = matrix.reduce((a, b) => merge(a, b))
  return arr[k - 1]
};

function merge(left, right){
  let llen = left.length
  let rlen = right.length
  let i = 0
  let j = 0
  let res = []
  // 后入数组先按大小入目标数组
  while(i < llen && j < rlen){
    if (left[i] < right[j]) {
      res.push(left[i++])
    } else {
      res.push(right[j++])
    }
  }
  // 排序逻辑中未入目标数组的子集依次进入
  while(i < llen) res.push(left[i++])
  while(j < rlen) res.push(right[j++])
  return res
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
探险家小扣的行动轨迹,都将保存在记录仪。expeditions[i] 表示小扣第 i 次探险记录,用一个字符串数组表示。其的每个「营地」由大小写字母组成,通过子串 -> 连接。例:"Leet->code->Campsite",表示到访了 "Leet"、"code"、"Campsite" 三个营地。expeditions[0] 包含了初始小扣已知的所有营地;对于之后的第 i 次探险(即 expeditions[i] 且 i > 0),如果记录包含了之前均没出现的营地,则表示小扣 新发现 的营地。 请你找出小扣发现新营地最多且索引最小的那次探险,并返回对应的记录索引。如果所有探险记录都没有发现新的营地,返回 -1。注意: 大小写不同的营地视为不同的营地; 营地的名称长度均大于 0。用python实现。给你几个例子:示例 1: 输入:expeditions = ["leet->code","leet->code->Campsite->Leet","leet->code->leet->courier"] 输出:1 解释: 初始已知的所有营地为 "leet" 和 "code" 第 1 次,到访了 "leet"、"code"、"Campsite"、"Leet",新发现营地 2 处:"Campsite"、"Leet" 第 2 次,到访了 "leet"、"code"、"courier",新发现营地 1 处:"courier" 第 1 次探险发现的新营地数量最多,因此返回 1。示例 2: 输入:expeditions = ["Alice->Dex","","Dex"] 输出:-1 解释: 初始已知的所有营地为 "Alice" 和 "Dex" 第 1 次,未到访任何营地; 第 2 次,到访了 "Dex",未新发现营地; 因为两次探险均未发现新的营地,返回 -1
04-23
我的理解是,你想让我编写一段 Python 代码,实现一个算法,用于在给定的探险记录找到发现新营地最多且索引最小的探险记录。 那么首先,我需要解析题目的输入输出要求。 输入要求: - expeditions:一个字符串数组,表示小扣第一次到访的所有营地,和之后的每次探险记录(一个记录就是一个字符串,其营地通过子串 -> 连接)。 输出要求: - 如果所有的探险记录都没有发现新的营地,则返回 -1; - 否则,返回发现新营地最多且索引最小的探险记录的索引。注:探险记录的索引从 0 开始计数。 接下来,我可以考虑如何编写代码来解决这个问题。 我的思路是,首先用一个集合记录所有已知的营地名称,然后遍历每个探险记录,依次将其的营地名称加入到集合,并判断是否有新的营地被发现。 如果某个探险记录发现了新的营地,那么我就记录下发现了多少个新营地,以及这个探险记录的索引。最后,遍历所有的探险记录,找到发现新营地最多且索引最小的记录。 代码如下: ```python def find_new_camp(expeditions): known_camps = set() # 已知的营地集合 max_new_camps = 0 # 最多的新营地数量 max_new_camps_index = -1 # 最多新营地数量的探险记录的索引 for i in range(len(expeditions)): sub_camps = expeditions[i].split("->") # 提取探险记录所有的营地名称 num_new_camps = 0 # 记录这个探险记录发现的新营地数量 for camp in sub_camps: if camp not in known_camps: # 如果这个营地是新的,就记录一下 known_camps.add(camp) num_new_camps += 1 if num_new_camps > max_new_camps: # 如果发现了更多的新营地,就更新最大值和索引 max_new_camps = num_new_camps max_new_camps_index = i return max_new_camps_index if max_new_camps > 0 else -1 ``` 我使用了两个变量 `max_new_camps` 和 `max_new_camps_index` 来记录已经遍历过的探险记录,发现了最多的新营地数量,以及具有最多新营地数量的探险记录的索引。然后我遍历所有的探险记录,统计其发现的新营地数量,如果大于当前的最大值,就更新 `max_new_camps` 和 `max_new_camps_index` 变量。最后,如果发现了新的营地,就返回 `max_new_camps_index`。如果所有的探险记录都没有发现新营地,就返回 -1。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值