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;
}

加粗样式

相关推荐
DC_BLOG1 分钟前
数据结构绪论
java·数据结构·算法
Wils0nEdwards1 小时前
Leetcode 加一
java·算法·leetcode
xuchaoxin13751 小时前
三角矩阵和对称阵的压缩存储
算法·矩阵
youk1101 小时前
STM32 HAL库之SDIO例程 Micro SD卡 - 1
算法
power-辰南1 小时前
机器学习支持向量机(SVM)算法
人工智能·python·算法·机器学习·支持向量机
空雲.2 小时前
牛客周赛71(字符串,状压dp)
数据结构·算法
橘子遇见BUG2 小时前
算法日记48 day 图论(拓扑排序,dijkstra)
算法·图论
s_m_c2 小时前
Any2Policy: Learning Visuomotor Policy with Any-Modality(类似AnyGPT)
人工智能·深度学习·算法·机器学习
南宫生2 小时前
力扣-图论-10【算法学习day.60】
java·学习·算法·leetcode·图论
手捧向日葵的话语2 小时前
图论 —— 求解最短路径(Dijkstra算法、Bellman-Ford算法、Floyd-Warshall算法)
数据结构·c++·算法·图论