leetcode 之 二分查找(java)(3)

文章目录

    • [5. 81. 搜索旋转排序数组 II](#5. 81. 搜索旋转排序数组 II)
    • [6. 378、有序矩阵中第k个小的元素](#6. 378、有序矩阵中第k个小的元素)

5. 81. 搜索旋转排序数组 II

题目描述

已知存在一个按非降序排列的整数数组 nums ,数组中的值不必互不相同。

在传递给函数之前,nums 在预先未知的某个下标 k0 <= k < nums.length)上进行了 旋转 ,使数组变为 [nums[k], nums[k+1], ..., nums[n-1], nums[0], nums[1], ..., nums[k-1]](下标 从 0 开始 计数)。例如, [0,1,2,4,4,4,5,6,6,7] 在下标 5 处经旋转后可能变为 [4,5,6,6,7,0,1,2,4,4]

给你 旋转后 的数组 nums 和一个整数 target ,请你编写一个函数来判断给定的目标值是否存在于数组中。如果 nums 中存在这个目标值 target ,则返回 true ,否则返回 false

你必须尽可能减少整个操作步骤。

示例 1:

复制代码
输入:nums = [2,5,6,0,0,1,2], target = 0
输出:true

示例 2:

复制代码
输入:nums = [2,5,6,0,0,1,2], target = 3
输出:false

提示:

  • 1 <= nums.length <= 5000
  • -104 <= nums[i] <= 104
  • 题目数据保证 nums 在预先未知的某个下标上进行了旋转
  • -104 <= target <= 104

思路

该题目与其上一道题目的区别是,该题目数组中有重复的数,因此,在判断的时候会有一些不同。

  • 例如,存在两端数都与mid处的数相同的情况。这种情况,我们只能将 l ++ , r --。因此,从代码中可以看出,该题目与上一个题目的不同之处就是,该题目添加了一个
  • if(nums[l] == nums[mid] && nums[mid] == nums[r]) {
    l ++;
    r --;
    } 的逻辑。
  • 同时,将判断有序数组的左右边界改成了 l 和 r。

代码

java 复制代码
int l = 0, r = n - 1;
while(l <= r)
{
    int mid = (l + r) / 2;
    if(nums[mid] == target) return true;
    if(nums[l] == nums[mid] && nums[mid] == nums[r]) {
        l ++;
        r --;
    } else if(nums[l] <= nums[mid]) {
        if(nums[l] <= target && target < nums[mid]) r = mid - 1;
        else l = mid + 1;
    } else {
        if(nums[mid] < target && target <= nums[r]) l = mid + 1;
        else r = mid - 1;
    }
}

6. 378、有序矩阵中第k个小的元素

题目描述

给你一个 n x n 矩阵 matrix ,其中每行和每列元素均按升序排序,找到矩阵中第 k 小的元素。

请注意,它是 排序后 的第 k 小元素,而不是第 k不同 的元素。

你必须找到一个内存复杂度优于 O(n2) 的解决方案。

示例 1:

复制代码
输入:matrix = [[1,5,9],[10,11,13],[12,13,15]], k = 8
输出:13
解释:矩阵中的元素为 [1,5,9,10,11,12,13,13,15],第 8 小元素是 13

示例 2:

复制代码
输入:matrix = [[-5]], k = 1
输出:-5

提示:

  • n == matrix.length
  • n == matrix[i].length
  • 1 <= n <= 300
  • -109 <= matrix[i][j] <= 109
  • 题目数据 保证 matrix 中的所有行和列都按 非递减顺序 排列
  • 1 <= k <= n2

题解

题目中说到,每行和每列元素均按升序排序,

  • 说明矩阵有一个性质,就是从左上到右下数组中的数值依次递减。
  • 我们遍历的时候,取左上最小值,右下最大值,然后,结果肯定在这两个的范围内。
  • 同时每次check()时,从左下开始遍历,因为从对角线分割,左下往右上遍历,每次统计每列小于mid的行数,统计完总数,再返回判断结果。
  • 如果总数大于等于k,那么说明mid大了,r = mid
  • 如果总数小于k,那么说明mid小了,l = mid + 1

代码

java 复制代码
public int kthSmallest(int[][] matrix, int k) {
    int n = matrix.length;
    int l = matrix[0][0], r = matrix[n - 1][n - 1];

    while(l < r) {
        int mid = l + (r - l) / 2;
        if(check(matrix, mid, k, n)) {
            r = mid;
        } else {
            l = mid + 1;
        }
    }
    return l;
}

public boolean check(int[][] matrix, int mid, int k, int n) {
    int i = n - 1, j = 0;
    int num = 0;
    while(i >= 0 && j < n)
    {
        if(matrix[i][j] <= mid) {
            num += i + 1;
            j ++;
        } else {
            i --;
        }
    }
    return num >= k;
}

加粗样式

相关推荐
木子.李34726 分钟前
排序算法总结(C++)
c++·算法·排序算法
闪电麦坤952 小时前
数据结构:递归的种类(Types of Recursion)
数据结构·算法
Gyoku Mint2 小时前
机器学习×第二卷:概念下篇——她不再只是模仿,而是开始决定怎么靠近你
人工智能·python·算法·机器学习·pandas·ai编程·matplotlib
纪元A梦2 小时前
分布式拜占庭容错算法——PBFT算法深度解析
java·分布式·算法
px不是xp3 小时前
山东大学算法设计与分析复习笔记
笔记·算法·贪心算法·动态规划·图搜索算法
枫景Maple4 小时前
LeetCode 2297. 跳跃游戏 VIII(中等)
算法·leetcode
鑫鑫向栄4 小时前
[蓝桥杯]修改数组
数据结构·c++·算法·蓝桥杯·动态规划
鑫鑫向栄4 小时前
[蓝桥杯]带分数
数据结构·c++·算法·职场和发展·蓝桥杯
小wanga4 小时前
【递归、搜索与回溯】专题三 穷举vs暴搜vs回溯vs剪枝
c++·算法·机器学习·剪枝