【面试经典 150 | 二分查找】搜索插入位置
文章目录
- 写在前面
- Tag
- 题目来源
- 题目解读
- 解题思路
- 方法一:二分查找
- 闭区间
- 左闭右开区间
- 开区间
- 总结
- 知识总结
- 写在最后
写在前面
本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更……
专栏内容以分析题目为主,并附带一些对于本题涉及到的数据结构等内容进行回顾与总结,文章结构大致如下,部分内容会有增删:
- Tag:介绍本题牵涉到的知识点、数据结构;
- 题目来源:贴上题目的链接,方便大家查找题目并完成练习;
- 题目解读:复述题目(确保自己真的理解题目意思),并强调一些题目重点信息;
- 解题思路:介绍一些解题思路,每种解题思路包括思路讲解、实现代码以及复杂度分析;
- 知识回忆:针对今天介绍的题目中的重点内容、数据结构进行回顾总结。
Tag
【二分查找】【数组】
题目来源
35. 搜索插入位置

题目解读
其实就是找出数组中大于等于 target 的最小值所在的位置。
解题思路
在数组中找到第一个大于或者等于 target 值的所在位置,数据量不大,约为 O ( 1 0 4 ) O(10^4) O(104),可以一次遍历查找,也开始使用二分查找来解决。题目要求使用时间复杂度为 O ( l o g n ) O(logn) O(logn) 的算法,那就只能使用二分法来解答了。
方法一:二分查找
二分查找是一种高效的搜索算法,在有序数组的指定区间内根据要求搜索元素,根据当前区间的中点的搜索情况缩短区间,每次搜索都会缩短一半的区间,时间复杂度为 O ( l o g n ) O(logn) O(logn), n n n 为有序数组的长度。
二分查找根据区间的开闭分为以下三种情况:
- 闭区间;
- 左闭右开区间;
- 开区间。
二分查找的区间无论是哪种闭合形式,要关注的 问题 始终是三个:
- 一是何时退出循环;
- 二是表示左、右区间的左、右指针如何移动;
- 三是返回什么。
以下将针对三种二分法的三个问题分别进行分析。
闭区间
闭区间表示搜索的区间是闭合的,初始的闭合区间为 [0, n-1]
,即指针指向 l = 0, r = n-1
。
何时退出循环?
当区间为空的时候退出 while 循环,何时区间为空?
当 l > r
时,指针 l
位于指针 r
右侧已经构不成区间了,退出 while 循环。
因此,此时的 while 循环条件为 l <= r
。
左、右指针如何移动?
在闭区间情况下,如果 nums[mid] < target
,说明 >= target
的值在右侧区间,于是只需要更新左指针 l = mid + 1
;否则(nums[mid] >= target
)说明 mid
及其之后的位置都是确定大于 target
的,这时候需要更新右指针 r = mid - 1
来判断左区间内的数。
返回什么?
退出循环后,r
指向的位置为 l
指向位置左侧的第一个位置,最后返回 l
或者 r + 1
。
实现代码
class Solution {
public:int searchInsert(vector<int>& nums, int target) {int n = nums.size();int l = 0, r = n - 1;while (l <= r) { // 区间为空退出循环int mid = l + ((r - l) >> 1);if (nums[mid] < target) {l = mid + 1;}else {r = mid - 1;}}return l; // 等价于 return r+1}
};
左闭右开区间
左闭右开区间表示搜索的区间是左闭右开的,初始的指针指向为 l = 0, r = n
。
何时退出循环?
当区间为空的时候退出 while,何时区间为空?
因为 r
指针是取不到的,所以当 l = r
时,区间为空(因为实际的区间左端点为 l,右端点为 r,构不成区间),退出 while 循环。
此时的 while 循环条件为 l < r
。
左、右指针如何移动?
在左闭右开区间情况下,如果 nums[mid] < target
,说明 >= target
的值在右侧区间,于是只需要更新左指针 l = mid + 1
;否则(nums[mid] >= target
)说明 mid
及其之后的位置都是确定大于 target
的,这时候需要更新右指针 r = mid
来判断左区间即 [l, mid-1]
中的数。
返回什么?
退出循环后,r
和 l
指向的是同一个位置,最后返回 l
或者 r
。
实现代码
class Solution {
public:int searchInsert(vector<int>& nums, int target) {int n = nums.size();int l = 0, r = n;while (l < r) { // 区间为空退出循环int mid = l + ((r - l) >> 1);if (nums[mid] < target) {l = mid + 1;}else {r = mid;}}return l; // 等价于 return r}
};
开区间
开区间表示搜索的区间两边都是开的,初始的指针指向为 l = -1, r = n
。
何时退出循环?
当区间为空的时候退出 while,何时区间为空?
因为 l
和 r
指针指向的值是取不到的,所以当 l + 1 = r
时,(l, r)
表示的区间为空,退出 while 循环。
此时的 while 循环条件为 l + 1 < r
。
左、右指针如何移动?
在开区间情况下,如果 nums[mid] < target
,说明 >= target
的值在右侧区间,于是只需要更新左指针 l = mid
(因为左区间是开的,实际的区间是 mid + 1
);否则(nums[mid] >= target
)说明 mid
及其之后的位置都是确定大于 target
的,这时候需要更新右指针 r = mid
来判断左区间即 [l, mid - 1]
中的数。
返回什么?
退出循环后,l
指向的位置位于 r
指向位置的左侧,最后返回 r
或者 l+1
。
总结
无论是哪种区间闭合形式,关注的始终是:何时退出while循环、左右指针如何更新以及最后返回什么这三个问题。
需要注意的移动指针一定要保证原来区间的闭合与否:原来区间是闭合的,更新后的指针指向的依旧是闭合的区间;原来区间是开的,更新后的指针指向的依旧是开的区间。
以上三种形式,可以挑选一种自己熟悉的形式记忆,并记住这是解决在有序数组中查找大于等于指定值的代码。
知识总结
在有序数组中查找大于等于指定值的最小位置的二分查找方法是十分经典,该方法已经被封装成了一个函数 lower_bound()
,该方法之所以是经典是因为其他类型的查找都可以通过它来完成:
- 在有序数组中查找大于等于指定值的最小位置,这就不赘述了;
- 在有序数组中查找大于指定值
x
的最小位置,本题及以下等价转换仅限于整数数组。此时可以等价转化为 “在有序数组中查找大于等于指定值x+1
的最小位置”; - 在有序数组中查找小于指定值
x
的最大位置,可以等价转化为 “在有序数组中查找大于等于指定值x
的最小位置” 减1
; - 在有序数组中查找小于等于指定值
x
的最大位置,可以等价转化为 “在有序数组中查找大于等于指定值x+1
的最小位置” 减1
。
写在最后
如果文章内容有任何错误或者您对文章有任何疑问,欢迎私信博主或者在评论区指出 💬💬💬。
如果大家有更优的时间、空间复杂度方法,欢迎评论区交流。
最后,感谢您的阅读,如果感到有所收获的话可以给博主点一个 👍 哦。
相关文章:

【面试经典 150 | 二分查找】搜索插入位置
文章目录 写在前面Tag题目来源题目解读解题思路方法一:二分查找闭区间左闭右开区间开区间总结 知识总结写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更…… 专栏内容以分析题目为主,…...

DAPP开发【06】nodejs安装与npm路径更换
windows系统在执行用户命令时顺序 windows系统在执行用户命令时,若用户未给出文件的绝对路径, 则 (1)首先在当前目录下寻找相应的可执行文件、批处理文件等; (2)若找不到,再依次在系…...

数据结构奇妙旅程之顺序表和链表
꒰˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好,我是xiaoxie.希望你看完之后,有不足之处请多多谅解,让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN …...

vitepress的使用
创建项目并启动项目 // 1.创建项目,直接在空项目下安装vitepress(npm/yarn等都可以,这个可以看官网,官网给了好几种安装方式) yarn add -D vitepress // 2.初始化配置项目(npm/官网也给了多种包管理工具的安装方式) npx vitepress init // 初始化命令执行完会遇到以下几个问题…...

Discuz论坛自动采集发布软件
随着网络时代的不断发展,Discuz论坛作为一个具有广泛用户基础的开源论坛系统,其采集全网文章的技术也日益受到关注。在这篇文章中,我们将专心分享通过输入关键词实现Discuz论坛的全网文章采集,同时探讨采集过程中伪原创的发布方法…...
B树在数据库的应用
B树(B-tree)是一种自平衡的树状数据结构,广泛应用于数据库和文件系统等领域,其设计的目标是提供一种高效的插入、删除和查找操作。B树的设计是为了在磁盘等存储介质上存储和操作大量的数据。 主要特点包括: 平衡性&a…...

Android 源码编译
一,虚拟机安装 1.1 进入https://cn.ubuntu.com/download中文官网下载iso镜像 1.2 这里我们下载Ubuntu 18.04 LTS 1.3虚拟VM机安装ubuntu系统,注意编译源码需要至少16G运行内存和400G磁盘空间,尽量设大点 二 配置编译环境 2.1 下载andr…...

信而泰 SSL测试方法介绍
[本文介绍在ALPS平台上进行SSL测试的内容和方法] 什么是SSL SSL全称是Secure Sockets Layer,指安全套接字协议,为基于TCP的应用层协议提供安全连接;SSL介于TCP/IP协议栈的第四层和第五层之间,广泛用于电子商务、网上银行等。 SSL…...

Redis--15--缓存穿透 击穿 雪崩
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 缓存穿透 击穿 雪崩运行速度:1 缓存穿透问题描述:如何解决: 2 缓存击穿问题描述:如何解决: 3 缓存雪崩说明:解决方案: 缓存穿透 击穿 雪崩 问题描述: 由于海量的用…...

excel表格在线编辑(开源版)
文章目录 前言一、Luckysheetvue3vite 例子如有启发,可点赞收藏哟~ 前言 本文记录好用的开源在线表格 具体如图显示 另外记录下更名后的univer~,如下图(有兴趣可自行详细了解) univer 在线思维导图 一、Luckysheet 参考git…...

17.字符串处理函数——字符串比较函数
文章目录 前言一、题目描述 二、解题 程序运行代码 总结 前言 本系列为字符串处理函数编程题,点滴成长,一起逆袭。 一、题目描述 二、解题 程序运行代码 #include<stdio.h> #include<string.h> int main() {char *str1 "hello wo…...

【面试HOT200】二叉树——深度优先搜索篇
系列综述: 💞目的:本系列是个人整理为了秋招面试的,整理期间苛求每个知识点,平衡理解简易度与深入程度。 🥰来源:材料主要源于【CodeTopHot200】进行的,每个知识点的修正和深入主要参…...
价值投资选股的方法
价值投资法是一种长期投资策略,其核心思想是寻找被市场低估的股票,即股票的市场价格低于其内在价值。这种策略认为,投资者应该关注公司的基本面,如盈利能力、成长潜力、财务状况等,而不是短期的市场波动。以下是价值投…...
java中如何将mysql里面的数据取出来然后通过stream流的方式进行数据处理代码实例?
在 Java 中使用 Stream 流的方式从 MySQL 数据库中取出数据并进行处理,你可以通过 JDBC(Java Database Connectivity)来实现。下面是一个简单的代码示例: import java.sql.*; import java.util.stream.Stream; public class MySQ…...
C++服务器 支持http、tcp protobuf、websocket,linux开源框架 零依赖轻松编译部署 Reactor
开源地址: https://github.com/crust-hub/tubekit/tree/main Github:https://github.com/gaowanlu 诚招有兴趣的小伙伴加入开发维护 Tubekit The C TCP server framework based on the Reactor model continues to implement POSIX thread pool, Epoll, non blocking IO, obj…...

1688API接口系列,1688开放平台接口使用方案(商品详情数据+搜索商品列表+商家订单类)
1688商品详情接口是指1688平台提供的API接口,用于获取商品详情信息。通过该接口,您可以获取到商品的详细信息,包括商品标题、价格、库存、描述、图片等。 要使用1688商品详情接口,您需要先申请1688的API权限,并获取ac…...

CentOS服务器网页版Rstudio-server及R包批量安装最佳实践
CentOS服务器安装网页版Rstudio-server及R包批量安装 以下为CentOS 7/8的Rstudio-server安装、配置和R包安装操作 1. 软件包安装 Centos 7安装 # 下载安装包,大小115.14 MB wget -c https://download2.rstudio.org/server/centos7/x86_64/rstudio-server-rhel-…...

centos7内核升级(k8s基础篇)
1.查看系统内核版本信息 uname -r 2.升级内核 2.1更新yum源仓库 yum -y update更新完成后,启用 ELRepo 仓库并安装ELRepo仓库的yum源 ELRepo 仓库是基于社区的用于企业级 Linux 仓库,提供对 RedHat Enterprise (RHEL) 和 其他基于 RHEL的 Linux 发行…...
数据结构与算法设计分析——NP完全理论
目录 一、P类问题与NP类问题的定义二、常见的NP类问题(一)旅行商问题(TSP)(二)哈密尔顿回路问题(三)判断回路问题(四)图的着色问题(五)…...

AGNES层次聚类
已知数据集D中有9个数据点,分别是(1,2),(2,3),(2,1), (3,1),(2,4),(3,5),(4,3),(1,5),(4,2)。要求: (1)采用层次聚类的聚集算法进行聚类,k2。 (2)距离计算采用欧几里得距离。 (3)簇之间的距离采用单链接方…...

Docker 离线安装指南
参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...

label-studio的使用教程(导入本地路径)
文章目录 1. 准备环境2. 脚本启动2.1 Windows2.2 Linux 3. 安装label-studio机器学习后端3.1 pip安装(推荐)3.2 GitHub仓库安装 4. 后端配置4.1 yolo环境4.2 引入后端模型4.3 修改脚本4.4 启动后端 5. 标注工程5.1 创建工程5.2 配置图片路径5.3 配置工程类型标签5.4 配置模型5.…...

盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...
解锁数据库简洁之道:FastAPI与SQLModel实战指南
在构建现代Web应用程序时,与数据库的交互无疑是核心环节。虽然传统的数据库操作方式(如直接编写SQL语句与psycopg2交互)赋予了我们精细的控制权,但在面对日益复杂的业务逻辑和快速迭代的需求时,这种方式的开发效率和可…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现
摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序,以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务,提供稳定高效的数据处理与业务逻辑支持;利用 uniapp 实现跨平台前…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)
🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

ios苹果系统,js 滑动屏幕、锚定无效
现象:window.addEventListener监听touch无效,划不动屏幕,但是代码逻辑都有执行到。 scrollIntoView也无效。 原因:这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作,从而会影响…...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...

一些实用的chrome扩展0x01
简介 浏览器扩展程序有助于自动化任务、查找隐藏的漏洞、隐藏自身痕迹。以下列出了一些必备扩展程序,无论是测试应用程序、搜寻漏洞还是收集情报,它们都能提升工作流程。 FoxyProxy 代理管理工具,此扩展简化了使用代理(如 Burp…...