LeetCode刷题笔记:二分查找简单进阶

简介: 对于二分算法的简单进阶认识,利用两道LeetCode题型巩固认识!

这里将对刷题笔记一文末提及的几道推荐二分法进阶题目进行说明介绍。一道简单题加了一定的文字修饰,一道中等题巧用二分查找,以下为刷题笔记一链接,题目链接在文末提供。

题目一:搜索插入位置

题目描述

给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

测试用例

示例 1:
输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:
输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:
输入: nums = [1,3,5,6], target = 7
输出: 4

示例 4:
输入: nums = [1,3,5,6], target = 0
输出: 0

示例 5:
输入: nums = [1], target = 0
输出: 0

提示:

  • 1 <= nums.length <= 104
  • -104 <= nums[i] <= 104
  • nums 为无重复元素的升序排列数组
  • -104 <= target <= 104

解题思路

由题目提示中目标数组为升序有序无重复元素的数组且题目要求为寻找目标值以及对于时间复杂度的要求,可以清楚地认识到本题使用二分法解决问题。

解题步骤:

  1. 对目标数组进行二分查找,找到目标值,返回对应下标,否则,进行下一步。
  2. 左右全闭:
    i. 对于目标值在数组所有元素之前 [0, -1]
    ii. 对于目标值插入数组中的位置 [left, right],return right + 1
    iii. 对于目标值在数组所有元素之后的情况 [left, right], return right + 1
  3. 左闭右开:
    i. 目标值在数组所有元素之前 [0,0)
    ii. 目标值插入数组中的位置 [left, right) ,return right 即可
    iii. 目标值在数组所有元素之后的情况 [left, right),return right 即可

Java程序代码:

  • 左右全闭
class Solution {
   
   
    public int searchInsert(int[] nums, int target) {
   
   
        int left = 0 ;
        int right = nums.length-1;
        int middle = left + (right-left)/2;
        if(nums[left]>target){
   
   
                return 0;
            }
        if(nums[right]<target){
   
   
            return right + 1;
        }
        while(left<=right){
   
   
            if(nums[middle]<target){
   
   
                left=middle+1;
                middle = left + (right-left)/2;
            }
            else if(nums[middle]>target){
   
   
                right=middle-1;
                middle = left + (right-left)/2;
            }
            else{
   
   
                return middle;
            }
        }

        return right+1;
    }
}
  • 左闭右开
class Solution {
   
   
    public int searchInsert(int[] nums, int target) {
   
   
        int left = 0 ;
        int right = nums.length;
        int middle = left + (right-left)/2;
        if(nums[left]>target){
   
   
                return 0;
            }
        if(nums[right]<target){
   
   
            return right;
        }
        while(left < right){
   
   
            if(nums[middle]<target){
   
   
                left=middle+1;
                middle = left + (right-left)/2;
            }
            else if(nums[middle]>target){
   
   
                right=middle-1;
                middle = left + (right-left)/2;
            }
            else{
   
   
                return middle;
            }
        }

        return right;
    }
}

题目二:在排序数组中查找元素的第一个和最后一个位置

题目描述

给定一个按照升序排列的整数数组 nums,和一个目标值 target。找出给定目标值在数组中的开始位置和结束位置。

如果数组中不存在目标值 target,返回 [-1, -1]。

你可以设计并实现时间复杂度为 O(log n) 的算法解决此问题吗?

测试用例

示例 1:
输入:nums = [5,7,7,8,8,10], target = 8
输出:[3,4]

示例 2:
输入:nums = [5,7,7,8,8,10], target = 6
输出:[-1,-1]

示例 3:
输入:nums = [], target = 0
输出:[-1,-1]

提示

  • 0 <= nums.length <= 105
  • -109 <= nums[i] <= 109
  • nums 是一个非递减数组
  • -109 <= target <= 109

解题思路一

题目要求的是取得目标值 target 出现的闭区间(包含且等于 target ),即左边界为第一次小于 target 值的下标加一,右边界为第一次大于目标值 target 的下标减一,如若为 {5,7,7,8,8,10}, target = 8 ,返回 {3,4}。

左右边界的获取方法可以说是一致,但需要判断方法有所不同:

  • 左边界:从左到右二分查找小于目标值 target 下标
  • 右边界:从左到右二分查找大于目标值 target 下标

根据题目要求寻找target在数组里的左右边界后,可分为如下几种情况:

  • target 在数组范围之外,即在数组范围的右边或者左边。
    例如数组 {3, 4, 5},target 为 2 或者数组 {3, 4, 5},target 为 6,此时应该返回 {-1, -1}

  • target 在数组范围中,分两种情况:tareget存在数组之中,不存在数组之中。如下:

    • 数组中不存在 target。
      例如数组 {3,6,7} ,target 为 5,此时应该返回 {-1, -1}

    • 数组中存在 target。左边界加一,右边界减一,得到区间。
      例如数组 {3,6,7} ,target 为 6,此时应该返回 {1, 1}

了解完基本思路,我们就要去实现代码来寻找左右边界,且如果对于二分法足够清晰,可以采用同时查找左右边界的方式,如果,还只是初学者,建议先从一个边界值下手,再来处理另一个边界值。

Java 程序代码

  • 分别寻找左右边界,再根据三种情况讨论结果。
class Solution {
   
   
    int[] searchRange(int[] nums, int target) {
   
   
        int leftBorder = getLeftBorder(nums, target);
        int rightBorder = getRightBorder(nums, target);
        // 数组中不存在 target。
        if (leftBorder == -2 || rightBorder == -2) {
   
   
                    return new int[]{
   
   -1, -1};
                }
        // 数组中存在 target
        if (rightBorder - leftBorder > 1) {
   
   
                    return new int[]{
   
   leftBorder + 1, rightBorder - 1};
                }
        // target 在数组范围之外
        return new int[]{
   
   -1, -1};
    }

    int getRightBorder(int[] nums, int target) {
   
   
        int left = 0;
        int right = nums.length - 1;
        int rightBorder = -2; // 记录一下rightBorder没有被赋值的情况
        while (left <= right) {
   
   
            int middle = left + ((right - left) / 2);
            if (nums[middle] > target) {
   
   
                right = middle - 1;
            } 
                        else {
   
    // 寻找右边界,nums[middle] == target的时候更新left
                left = middle + 1;
                rightBorder = left;
            }
        }
        return rightBorder;
    }

    int getLeftBorder(int[] nums, int target) {
   
   
        int left = 0;
        int right = nums.length - 1;
        int leftBorder = -2; // 记录一下leftBorder没有被赋值的情况
        while (left <= right) {
   
   
            int middle = left + ((right - left) / 2);
            if (nums[middle] >= target) {
   
    // 寻找左边界,nums[middle] == target的时候更新right
                right = middle - 1;
                leftBorder = right;
            } 
                        else {
   
   
                left = middle + 1;
            }
        }
        return leftBorder;
    }
}

解题思路二

  • 先采用二分法寻找 target 值在数组是否存在。
  • 如存在,分别将 left 与 right 赋值为中间值,在分别移动判断是否存在重复值,直至不存在重复值得到下标返回 {left,right} 。
  • 不存在,返回 {-1,-1}

注意:

  • 左右边界值移动时,切忌数组下标越界,即下标小于零或大于数组长度-1,必须先判断下标不越界,才能对比值,顺序不可调换。对于条件的选择顺序应为:

    • left - 1 >= 0 && nums[left - 1] == nums[index]

    • right + 1 < nums.length && nums[right + 1] == nums[index]

class Solution {
   
   
    public int[] searchRange(int[] nums, int target) {
   
   
        int index = binarySearch(nums, target); // 二分查找

        if (index == -1) {
   
    // nums 中不存在 target,直接返回 {-1, -1}
            return new int[] {
   
   -1, -1}; // 匿名数组 
        }
        // nums 中存在 targe,则左右滑动指针,来找到符合题意的区间
        int left = index;
        int right = index;
        // 向左滑动,找左边界
        while (left - 1 >= 0 && nums[left - 1] == nums[index]) {
   
    // 防止数组越界。逻辑短路,两个条件顺序不能换
            left--;
        }
        // 向左滑动,找右边界
        while (right + 1 < nums.length && nums[right + 1] == nums[index]) {
   
    // 防止数组越界。
            right++;
        }
        return new int[] {
   
   left, right};
    }
    public int binarySearch(int[] nums, int target) {
   
   
        int left = 0;
        int right = nums.length - 1; // 不变量:左闭右闭区间

        while (left <= right) {
   
    // 不变量:左闭右闭区间
            int mid = left + (right - left) / 2;
            if (nums[mid] == target) {
   
   
                return mid;
            } else if (nums[mid] < target) {
   
   
                left = mid + 1;
            } else {
   
   
                right = mid - 1; // 不变量:左闭右闭区间
            }
        }
        return -1; // 不存在
    }
}

相关链接

相关题目链接

参考文章

求点赞转发

目录
相关文章
|
4月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
56 1
|
4月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
3月前
【LeetCode 01】二分查找总结
【LeetCode 01】二分查找总结
17 0
|
5月前
|
Python
【Leetcode刷题Python】50. Pow(x, n)
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
34 1
|
5月前
|
Python
【Leetcode刷题Python】1467. 两个盒子中球的颜色数相同的概率
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
56 0
|
5月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
64 6
|
5月前
|
Python
【Leetcode刷题Python】剑指 Offer 26. 树的子结构
这篇文章提供了解决LeetCode上"剑指Offer 26. 树的子结构"问题的Python代码实现和解析,判断一棵树B是否是另一棵树A的子结构。
58 4
|
5月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
130 2
|
5月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
77 7