【算法面试】在排序数组中查找元素的第一个和最后一个位置:详细题解

目录

题目描述

示例

示例 1:

示例 2:

示例 3:

问题分析

详细步骤

解决方法

方法 1:标准二分查找(分开查找第一个和最后一个)

方法 2:优化版二分查找(合并查找逻辑)

方法 3:带调试信息的版本

总结

博主v:XiaoMing_Java


题目描述

在处理有序数组时,一个常见的问题是查找特定元素的起始和结束位置。给定一个按照非递减顺序排列的整数数组 nums 和一个目标值 target,我们需要找到目标值在数组中的开始位置和结束位置。如果数组中不存在目标值 target,返回 [-1, -1]

示例

示例 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]

问题分析

由于数组是有序的,因此可以利用二分查找的方法来高效地解决这个问题。二分查找的时间复杂度为 O(log n),非常适合查找任务。

为了找到目标值的起始和结束位置,我们可以分别使用两次二分查找:

  1. 第一次查找目标值的起始位置。
  2. 第二次查找目标值的结束位置。

详细步骤

  1. 初始化:设置 left 为数组的起始索引 0right 为数组的末尾索引 n-1
  2. 查找起始位置
    • 使用二分查找,如果找到目标值,则继续向左半部分查找,直到找到最左边的目标值。
  3. 查找结束位置
    • 使用二分查找,如果找到目标值,则继续向右半部分查找,直到找到最右边的目标值。
  4. 组合结果:返回找到的起始位置和结束位置。

解决方法

方法 1:标准二分查找(分开查找第一个和最后一个)

以下是实现二分查找的一种标准方法,通过两次二分查找分别查找第一个和最后一个目标值的位置:

// 两次二分查找,分开查找第一个和最后一个
// 时间复杂度 O(log n), 空间复杂度 O(1)
// [1,2,3,3,3,3,4,5,9]
public int[] searchRange2(int[] nums, int target) {int left = 0;int right = nums.length - 1;int first = -1;int last = -1;// 找第一个等于target的位置while (left <= right) {int middle = (left + right) / 2;if (nums[middle] == target) {first = middle;right = middle - 1; // 重点} else if (nums[middle] > target) {right = middle - 1;} else {left = middle + 1;}}// 最后一个等于target的位置left = 0;right = nums.length - 1;while (left <= right) {int middle = (left + right) / 2;if (nums[middle] == target) {last = middle;left = middle + 1; // 重点} else if (nums[middle] > target) {right = middle - 1;} else {left = middle + 1;}}return new int[] { first, last };
}

方法 2:优化版二分查找(合并查找逻辑)

为了代码更简洁,可以将查找最左和最右位置的逻辑合并到一个函数中,并通过传递额外参数来控制查找方向:

public int[] searchRange(int[] nums, int target) {int first = findBound(nums, target, true);int last = findBound(nums, target, false);return new int[] { first, last };
}private int findBound(int[] nums, int target, boolean isFirst) {int left = 0;int right = nums.length - 1;int bound = -1;while (left <= right) {int middle = left + (right - left) / 2;if (nums[middle] == target) {bound = middle;if (isFirst) {right = middle - 1; // 查找最左位置} else {left = middle + 1; // 查找最右位置}} else if (nums[middle] > target) {right = middle - 1;} else {left = middle + 1;}}return bound;
}

方法 3:带调试信息的版本

为了更好地理解算法,可以添加调试信息来跟踪程序执行过程:

public int[] searchRange(int[] nums, int target) {int first = findBound(nums, target, true);int last = findBound(nums, target, false);return new int[] { first, last };
}private int findBound(int[] nums, int target, boolean isFirst) {int left = 0;int right = nums.length - 1;int bound = -1;while (left <= right) {int middle = left + (right - left) / 2;System.out.println("Left: " + left + ", Mid: " + middle + ", Right: " + right);if (nums[middle] == target) {bound = middle;if (isFirst) {right = middle - 1; // 查找最左位置} else {left = middle + 1; // 查找最右位置}} else if (nums[middle] > target) {right = middle - 1;} else {left = middle + 1;}}System.out.println(isFirst ? "First bound found at: " + bound : "Last bound found at: " + bound);return bound;
}

总结

通过使用二分查找的方式,我们可以在 O(log n) 的时间复杂度内高效地找到排序数组中目标值的起始和结束位置。本文展示了多种实现方法,包括标准二分查找、优化版二分查找以及带调试信息的版本,以帮助读者更好地理解并应用这种算法。希望这些方法能够帮助你解决实际开发中的问题,确保程序顺利运行。

 如果本文对你有帮助 欢迎 关注、点赞、收藏、评论!!!

博主v:XiaoMing_Java

 📫作者简介:嗨,大家好,我是 小 明(小明java问道之路),互联网大厂后端研发专家,2022博客之星TOP3 / 博客专家 / CSDN后端内容合伙人、InfoQ(极客时间)签约作者、阿里云签约博主、全网5万粉丝博主。


🍅 文末获取联系 🍅  👇🏻 精彩专栏推荐订阅收藏 👇🏻

专栏系列(点击解锁)

学习路线(点击解锁)

知识定位

🔥Redis从入门到精通与实战🔥

Redis从入门到精通与实战

围绕原理源码讲解Redis面试知识点与实战

🔥MySQL从入门到精通🔥

MySQL从入门到精通

全面讲解MySQL知识与企业级MySQL实战

🔥计算机底层原理🔥

深入理解计算机系统CSAPP

以深入理解计算机系统为基石,构件计算机体系和计算机思维

Linux内核源码解析

围绕Linux内核讲解计算机底层原理与并发

🔥数据结构与企业题库精讲🔥

数据结构与企业题库精讲

结合工作经验深入浅出,适合各层次,笔试面试算法题精讲

🔥互联网架构分析与实战🔥

企业系统架构分析实践与落地

行业最前沿视角,专注于技术架构升级路线、架构实践

互联网企业防资损实践

互联网金融公司的防资损方法论、代码与实践

🔥Java全栈白宝书🔥

精通Java8与函数式编程

本专栏以实战为基础,逐步深入Java8以及未来的编程模式

深入理解JVM

详细介绍内存区域、字节码、方法底层,类加载和GC等知识

深入理解高并发编程

深入Liunx内核、汇编、C++全方位理解并发编程

Spring源码分析

Spring核心七IOC/AOP等源码分析

MyBatis源码分析

MyBatis核心源码分析

Java核心技术

只讲Java核心技术

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

热电发电机越来越受到研发关注

热电发电机 (TEG) 利用热量&#xff08;或更准确地说&#xff0c;温差&#xff09;和众所周知的塞贝克效应来发电。它们的应用范围从收集可用热能&#xff0c;尤其是在工业和其他情况下“浪费”的热能&#xff0c;到在放射性同位素热发电机 (RTG) 中使用航天器的放射性电源作为…

【docker】2. 编排容器技术发展史(了解)

该篇文章介绍的主要是编排以及容器技术的发展史(了解即可)&#xff0c;如果想单纯学习docker命令操作可直接略过&#xff01;&#xff01;&#xff01; 容器技术发展史 Jail 时代 容器不是一个新概念或者新技术&#xff0c;很早就有了&#xff0c;只是近几年遇到了云计算&am…

mac单机游戏推荐:帝国时代2 for Mac中文安装包

帝国时代2是一款经典的即时战略游戏&#xff0c;由微软游戏工作室开发。玩家可以选择从欧洲、亚洲或美洲的不同文明开始游戏&#xff0c;建立自己的帝国并与其他文明进行战争、贸易和外交。游戏中包含了丰富的历史背景和真实的历史事件&#xff0c;玩家需要策略性地管理资源、招…

【机器学习】ChatTTS:开源文本转语音(text-to-speech)大模型天花板

目录 一、引言 二、TTS&#xff08;text-to-speech&#xff09;模型原理 2.1 VITS 模型架构 2.2 VITS 模型训练 2.3 VITS 模型推理 三、ChatTTS 模型实战 3.1 ChatTTS 简介 3.2 ChatTTS 亮点 3.3 ChatTTS 数据集 3.4 ChatTTS 部署 3.4.1 创建conda环境 3.4.2 拉取源…

Java项目毕业设计:基于springboot+vue的幼儿园管理系统

数据库:MYSQL5.7 **应用服务:Tomcat7/Tomcat8 使用框架springbootvue** 项目介绍 管理员&#xff1b;首页、个人中心、用户管理、教师管理、幼儿信息管理、班级信息管理、工作日志管理、会议记录管理、待办事项管理、职工考核管理、请假信息管理、缴费信息管理、幼儿请假管理…

【漏洞复现】极限OA video_file.php 任意文件读取漏洞

免责声明&#xff1a; 本文内容旨在提供有关特定漏洞或安全漏洞的信息&#xff0c;以帮助用户更好地了解可能存在的风险。公布此类信息的目的在于促进网络安全意识和技术进步&#xff0c;并非出于任何恶意目的。阅读者应该明白&#xff0c;在利用本文提到的漏洞信息或进行相关测…

开源最强AI一键抠图一键去背景模型RMBG-1.4

目录 前言训练数据图像分布&#xff1a;定性评估 应用场景安装用法 前言 RMBG v1.4 是BRIA AI最先进的背景去除模型&#xff0c;旨在有效地将各种类别和图像类型的前景与背景分开。该模型已在精心选择的数据集上进行训练&#xff0c;其中包括&#xff1a;一般库存图片、电子商…

使用 scikit-learn进行简单回归分析

使用 scikit-learn&#xff08;简称 sklearn&#xff09;进行简单回归分析的教程。这里我们以波士顿房价数据集&#xff08;Boston House Prices dataset&#xff09;为例&#xff0c;介绍如何使用线性回归&#xff08;Linear Regression&#xff09;模型 环境准备 确保已经安…

疲劳驾驶智能识别摄像机

疲劳驾驶智能识别摄像机在道路安全管理中扮演着越来越重要的角色。这些先进的设备不仅仅是简单的监控工具&#xff0c;它们通过先进的技术和算法&#xff0c;有效地识别和预防司机疲劳驾驶&#xff0c;从而大大减少了交通事故的风险。 首先&#xff0c;这些智能识别摄像机采用高…

IP地址的数量限制

IP地址&#xff08;IPv4&#xff09;是一个4字节32位的正整数&#xff0c;因此一共有个IP地址&#xff0c;即将近43亿个IP地址。但TCP/IP协议规定&#xff0c;每个主机都要有一个IP地址 现在全世界人口已经有70多亿了&#xff0c;电脑、手机等都需要IP地址。随着科技的发展&am…

如何通过数据驱动的动态调度来优化pcdn的效率?

要通过数据驱动的动态调度来优化P2P内容分发网络&#xff08;PCDN&#xff09;的效率&#xff0c;可以采取以下几个关键步骤&#xff1a; 一&#xff0e;数据收集与分析 1.流量监控&#xff1a;收集PCDN网络中每个节点的流量数据&#xff0c;包括上传、下载、转发量等。 2.性…

Android ANR日志分析指南

Android ANR日志分析指南 定义ANR类型ANR出现的原因ANR案例分析过程一、 查看events_log二、 traces.txt 日志分析 ANR 案例整理参考 定义 ANR(Application Not Responding) 应用程序无响应。如果你应用程序在UI线程被阻塞太长时间&#xff0c;就会出现ANR&#xff0c;通常出现…

2024年希望杯数学竞赛各年级100道练习题及答案

链接里面有无答案版本链接&#xff1a;https://pan.baidu.com/s/1nTIVJrTEWUzb0LJNo4mI_Q 提取码&#xff1a;0548 –来自百度网盘超级会员V7的分享 一年级 二年级 三年级 四年级 五年级 六年级 七年级 八年级

【博士每天一篇文献-算法】Fearnet Brain-inspired model for incremental learning

阅读时间&#xff1a;2023-12-16 1 介绍 年份&#xff1a;2017 作者&#xff1a;Ronald Kemker&#xff0c;美国太空部队&#xff1b;Christopher Kanan&#xff0c;罗切斯特大学 期刊&#xff1a; arXiv preprint 引用量&#xff1a;520 Kemker R, Kanan C. Fearnet: Brain-…

基于Java中的SSM框架实现萌宠优购系统项目【项目源码】计算机毕业设计

基于Java中的SSM框架实现萌宠优购系统演示 研究背景 现在人们在生活中、工作中压力与日俱增&#xff0c;而宠物也在人们的生活扮演者越来越重要的角色。然而&#xff0c;对于一些宠物爱好者来说&#xff0c;宠物早已经是他们生活中不可或缺的一部分。因为宠物能与人有心灵的沟…

为element-ui 打个补丁修复el-form的label width问题

为element-ui 打个补丁修复el-form的label width问题 问题回顾 用 element-ui 遇到一个奇怪的问题,控制台老是抛错 [Vue warn]: Error in beforeDestroy hook: "Error: (ElementForm]unpected width 最终排查下来是 el-form 设置了 label-width"auto" 然后又处…

Python实践项目 - 猜数字

实践项目 为了巩固所学知识并培养实践能力&#xff0c;我们可以设计一些有趣的实践项目。以下是一个简单的实践项目示例&#xff1a; 猜数字游戏 项目描述&#xff1a;编写一个猜数字游戏&#xff0c;程序随机生成一个1到100之间的整数&#xff0c;让玩家输入猜测的数字。如…

计算机进入安全模式

最近老是蓝屏------显示是REGISTRY_ERROR&#xff08;注册表错误&#xff09; 重启之后选择进入安全模式 我重启之后&#xff0c;按F8进不了安全模式。通过另一种方式进入&#xff0c;按住Shift键再按重启 重启之后出现以下界面 选择疑难解答&#xff0c;再选择高级选项 在高…

pandas保存成CSV格式时数据换行:SQL的REPLACE函数过滤掉数据的回车符

在使用Pandas保存数据到CSV文件时&#xff0c;如果数据中包含换行符&#xff08;例如\n&#xff09;&#xff0c;这可能会导致数据在CSV文件中被分割成多行&#xff0c;影响数据的完整性和可读性。为了解决这个问题&#xff0c;你可以在保存CSV之前使用Pandas的replace函数来替…

【华为OD机试】日志限流(C++/Java/Python)

题目 题目描述 某软件系统会在运行过程中持续产生日志,系统每天运行N单位时间,运行期间每单位时间产生的日志条数保行在数组records中。records[i]表示第i单位时间内产生日志条数。 由于系统磁盘空间限制,每天可记录保存的日志总数上限为total条。 如果一天产生的日志总条数…