算法与数据结构——二分查找插入点

news/2024/9/28 17:34:45

二分查找插入点

二分查找不仅可用于搜索目标元素,还可以解决许多变种问题,比如搜索目标元素的插入位置。

无重复元素情况

Question
给定一个长度为n的有序数组nums和一个元素target,数组不存在重复元素。现将target插入数组nums中,并保持其有序性。若数组中已存在元素target,则插入到其左方。请返回插入后target在数组中的索引。

问题一:当数组中包含target时,插入点的索引是否就是该元素的索引?

题目要求将target插入到相等元素的左边,这意味着插入的target替换了原来target的位置。也就是说,当数组包含target时,插入点的索引就是该target的索引

问题二:当数组中不存在target时,插入点是哪个元素的索引?

进一步思考二分查找过程(m为中点索引):当nums[m] < target时,这意味着指针i在向大于等于target的元素靠近。同理,指针j始终在向小于等于target的元素靠近。
因此二分结束时一定有:i指向首个大于target的元素,j指向首个小于target的元素。易得当数组不包含target时,插入索引为i

代码示例如下:

/*二分查找插入点(无重复元素)*/
int binarySearchInsertionSimple(vector<int> &nums, int target){int i = 0, j = nums.size() - 1;while (i <= j){int m = i + (j - i) / 2;if (nums[m] < target)i = m + 1;else if (nums[m] > target)j = m - 1;elsereturn m;  // 找到target 返回插入点 m}return i;			// 未找到 target,返回插入点 i
}

存在重复元素的情况

假设数组中存在多个target,则普通二分查找只能返回其中一个target的索引,而无法确定该元素的左边和右边还有多少个target

题目要求将目标元素插入到最左边,所以我们需要查找数组中最左一个target的索引

实现步骤:

  1. 执行二分查找,得到任意一个target索引,记为k。
  2. 从索引k开始,向左进行线性遍历,当找到最左边的target时返回。

此方法虽然可用,但其包含线性查找,因此时间复杂度为O(n)。当数组中存在很多重复的target时,该方法效率很低。

现考虑拓展二分查找代码。如图所示,整体流程保持不变,每轮现计算中点索引m,再判断targetnums[m]的大小关系,分以下几种情况:

  • nums[m] < targetnums[m] > target时,说明还没有找到target,因此采用普通二分查找的缩小区间操作,从而使指针i和j向target靠近
  • nums[m] == target时,说明小于target的元素在区间[i,m-1]中,因此采用j = m-1来缩小区间,从而使指针j向小于target的元素靠近

循环完成后,i指向最左边的target,j指向首个小于target的元素,因此索引i就是插入点








观察以下代码,其中判断分支nums[m] > targetnums[m] == target的操作相同,因此两者可以合并。即便如此,我们仍然可以将判断条件保持展开,因为逻辑更加清晰、可读性更好。

/*二分查找插入点(存在重复元素)*/
int binarySearchInsertion(vector<int> &nums, int target){int i = 0, j = nums.size() - 1;while (i <= j){int m = i + (j - 1) / 2;if (nums[m] < target)i = m + 1;		// target 在区间 [m+1, j] 中else if (nums[m] > target)j = m - 1;		// target 在区间 [i, m-1] 中elsej = m - 1;		// 首个小于 target 的元素在区间 [i, m-1] 中}// 返回插入点return i;
}

总的看来,二分查找无非就是给指针i和j分别设定搜索目标,目标可能是一个具体元素(例如target),也可能是一个元素范围(如小于target的元素)。

在不断的循环二分中,指针i和j都逐渐逼近预先设定的目标。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.ryyt.cn/news/59508.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈,一经查实,立即删除!

相关文章

2022 CSP-J 阅读程序3

1 2022 CSP-J 阅读程序3 阅读程序(程序输入不超过数组或字符串定义的范围;判断题正确填 √,错误填;除特 殊说明外,判断题 1.5 分,选择题 3 分) 源代码 #include<iostream>using namespace std;int n,k;int solve1() {int l=0,r=n;while(l<=r){int mid=(l+r)/2;…

第二届熵密杯-广外女生青春版

晨曦初始谜题1 由源码可知,有固定的前缀,且长度为18,超过一个块的长度,可以通过求方程的形式先将key求出来,再将整个key带入解密函数得到加密前的字符串 求key # sage N_HEX = "FFFFFFFEFFFFFFFFFFFFFFFFFFFFFFFF7203DF6B21C6052B53BBF40939D54123" N = Integer…

测评通义灵码,如何实现微信表情、 AI 语音笔记等小功能?

墨问西东是一家创业公司,很难一下子配齐像大公司那样的研发团队,这类 AI 编程辅助工具其实在一定程度上帮助我们的研发同学成长为全栈工程师,一个人就能顶上一个团队。内容来源 MacTalk 公众号,作者池建强 墨问西东是一家创业公司,很难一下子配齐像大公司那样的研发团队,…

一文看懂Prometheus告警原理及过程

本文详细介绍了如何在Prometheus中自定义告警规则,包括规则构成、Prometheus配置、告警流程以及告警解除的处理方法,特别关注了告警解除后的通知策略。摘要由CSDN通过智能技术生成目录 1. 自定义告警规则 2. 告警规则编写 3. prometheus配置 4. 告警过程 5. 告警解除 5.1 对s…

pytorch安装: cuda、cudatoolkit、torch版本对照

在 PyTorch 官网上有如下安装对照表,同时也有历史版本安装对照表从零开始配置python深度学习环境大概有如下配置步骤: 方案一: 电脑安装显卡驱动,然后安装CUDA、cuDNN,安装miniconda3。前面都是在电脑基础环境配置,后面的操作都是在conda环境中,安装torch、cudatoolkits…

AI实战 | 领克汽车线上营销助手:全面功能展示与效果分析

本篇文章的主要目的是为大家提供实现思路,以及如何更好地开发一个助手,而不仅仅是简单地进行拆解。如果采取拆解的方式,一篇文章可能会长达2万+字,还需要配以数十张图片,这将会非常繁琐。因此,针对拆解的详细内容,我计划单独制作一期视频,以帮助大家更清晰地理解。感谢…

ThreadLocal源码分析-

ThreadLocal源码分析 ThreadLocal是解决线程安全问题的一种方法,它通过为每个线程提供一个独立的变量副本避免了变量并发访问的冲突问题。一个ThreadLocal变量只与当前自身线程相关,对其他线程是隔离的。下面这段代码展示了ThreadLocal的使用。 public class test {private s…

SpringBoot源码分析

Springboot源码分析 1、SpringApplication初始化 从run()方法进入,可以看到Springboot首先创建了SpringApplication,然后调用SpringApplication的run()方法。 public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {return (…