【Java算法专场】二分查找(下)

目录

山脉数组的峰顶索引

算法分析 

算法步骤

算法代码

算法示例 

寻找峰值 

算法分析

算法步骤

算法代码

算法示例 

寻找旋转排序数组中的最小值 

算法分析

算法步骤

算法代码

算法示例 

点名 

算法分析

算法步骤

算法代码

算法示例 


山脉数组的峰顶索引

算法分析 

本道题是要在一个先升后降的数组查找拐点,如果我们采用暴力搜索的话,时间复杂度为O(n),但题目要求我们设计时间复杂度为O(logn)的算法,我们不难看出这道题也是具有二段性的,那么我们就可以采用二分查找。

算法步骤

  1. 初始化:设置双指针left和right,初始化left为1,right为nums.length-2.(由于要找峰顶,那么数组的第一个数和倒数第一个数必然不是,那么我们可以从第二个数出发)。
  2. 比较查找:定义mid,mid=left+(right-left+1)/2(为什么要+1,是为了避免整数溢出),通过与前一个值比较,判断当前位置是处于上坡还是下坡。当nums[mid]>mid[mid-1],说明此时mid处于上坡,让left=mid,当nums[mid]<=nums[mid-1]说明此时在下坡,让right=mid-1.循环条件为:left<right(当left==right时,说明找到了峰顶)。
  3. 返回结果:当循环结束之后,此时返回left即可。

算法代码

/*** 在山形数组中找到峰值的索引。* 山形数组是指一个数组,其中存在一个元素,它大于其前后的所有元素(即峰值),* 且该元素前后的元素递增或递减。** @param arr 山形数组,不为空且长度至少为3。* @return 返回数组中峰值的索引。如果有多个峰值,返回任意一个即可。*/public int peakIndexInMountainArray(int[] arr) {/* 初始化左右指针 */int left = 1, right = arr.length - 2;/* 使用二分查找法寻找峰值 */while (left < right) {/* 计算中间索引,避免整数溢出,并确保mid大于left */int mid = left + (right - left + 1) / 2;/* 如果中间元素大于其前一个元素,则峰值在mid或其右侧 */if (arr[mid] > arr[mid - 1]) {left = mid;} else {/* 否则,峰值在mid或其左侧 */right = mid - 1;}}/* 当左右指针相遇时,即找到峰值 */return left;}

时间复杂度为O(logn),n为数组的长度,每次循环都会干掉一半的数据。 

空间复杂度为O(1),只用了常数个变量。

算法示例 

以arr = [0,10,5,2]为例

第一步:初始化

left=1,right=2

第二步:查找峰顶

  1. mid=left+(right-left+1)/2=1+(2-1+1)/2=2,arr[mid]=5<arr[mid-1]=10,让right=mid-1=1.此时left=right=1,说明找到了峰顶。

第三步:返回结果

此时left=1,将left返回即可。  

寻找峰值 

算法分析

本道题与前面的题目类似,在数组中存在多个峰值,只需要找到其中一个峰值即可。若使用暴力解法,时间复杂度为O(n),我们可以使用二分查找,来降低复杂度,通过二分,能让时间复杂度达到O(logn).

算法步骤

  1. 初始化:定义双指针left和right,并初始化left为0,right为nums.length-1。
  2. 查找峰值:在eft和right之间进行循环。定义mid,mid=left+(right-left)/2,通过与后一个值比较,判断当前所处位置。当nums[mid]>nums[mid+1],说明此时在[left,mid]之间必然存在一个峰值,为什么能这样说?题目给我们说明了两边都是负无穷开始的,所以在[left,mid]中一定有峰值,让right=mid;当nums[mid]<=nums[mid+1]时,让left=mid+1即可,说明此时在[mid,left]中存在着峰值。循环条件为:left<right(当left和right相遇时,说明找到了峰值)
  3. 返回结果:将left返回即可。

算法代码

/*** 寻找峰值元素的索引。* 峰值元素被定义为大于其邻居的元素。* * @param nums 整数数组,其中存在至少一个峰值元素* @return 返回峰值元素的索引*/public int findPeakElement(int[] nums) {/* 初始化左右指针 */int left = 0;int right = nums.length - 1;/* 使用二分查找法寻找峰值 */while (left < right) {/* 计算中间索引,避免整数溢出 */int mid = left + (right - left) / 2;/* 如果中间元素大于其右侧元素,则峰值在左侧或就是中间元素 */if (nums[mid] > nums[mid + 1]) {right = mid;} else {/* 否则,峰值在右侧 */left = mid + 1;}}/* 当左右指针相遇时,即找到峰值 */return left;}

时间复杂度为O(logn),n为数组长度,在循环的过程中,每次都能排除掉一半的数据量。

空间复杂度为O(1),只用了常数个变量。 

算法示例 

 以nums=[1,2,1,3,5,6,4]

第一步:初始化

让left=0,right=6

第二步:查找峰值

  1. mid=left+(right-left)/2=0+(6-0)/2=3,  nums[mid]=3<nums[mid+1]=5,此时说明峰值可能在右区间,让left=mid+1=4

 2.mid=4+(6-4)/2=5,nums[mid]=6>nums[mid+1]=4,说明此时峰值在左区间,让right=mid。

3.mid=4+(5-4)/2=4,nums[mid]=5<nums[mid+1]=6,此时让left=mid+1,同时left和right相遇,说明此时找到了峰值.

 

第三步:返回结果

此时left=5,返回即可。 

寻找旋转排序数组中的最小值 

 

算法分析

本道题是要在一个被旋转后的数组中找到最小的元素,采用暴力遍历的算法,时间复杂度能达到O(n),但题目要求我们设计O(logn)的算法,我们可以看得出本道题也是具有二段性的,可以看成两段升序的数组。

算法步骤

  1. 初始化:设置双指针left和right,初始化left=0,right=nums.length-1。
  2. 预处理:如果数组在旋转之后还是原来的顺序,那么我们可以直接返回第一个元素,所以可以判断一下数组末尾的元素是否大于数组起始位置的元素,若大于则直接返回,反之则继续进行下续操作。
  3. 查找最小值:定义mid,mid=left+(right-left)/2,我们可以与数组两边的数进行比较,这里采用数组最右侧的数。当nums[mid]>mid[right],说明此时最小值在右区间[mid,right]中,让left=mid+1,当nums[mid]<=nums[right]时,说明此时最小值在左区间[left,mid],让right=mid,为什么不是right=mid-1,因为mid此时的位置有可能是最小值。循环条件为:left<right(当left和right相遇时,说明找到了最小值)
  4. 返回结果:返回nums[left]或nums[right]即可。

算法代码

/*** 在旋转后的有序数组中查找最小值。* 旋转有序数组是指原数组为非递减数组,将数组从某个位置分割成两部分,然后将两部分的顺序调换后形成的数组。* 例如,原数组[0,1,2,4,5,6,7]在数字4处旋转后变为[4,5,6,7,0,1,2]。* 此函数旨在在这种旋转后的数组中找到最小的数字。** @param nums 旋转后的有序数组* @return 数组中的最小值*/
public int findMin(int[] nums) {int left = 0, right = nums.length - 1;// 如果数组的第一个元素小于最后一个元素,说明数组未旋转或旋转后的最小值就是第一个元素if (nums[left] < nums[right]) {return nums[left];}// 使用二分查找法寻找最小值while (left < right) {int mid = left + (right - left) / 2;// 如果中间位置的元素大于最右边的元素,说明最小值在mid右侧if (nums[mid] > nums[right]) {left = mid + 1;} else {// 否则,最小值在mid或其左侧right = mid;}}// 最终right指向最小值的位置return nums[right];
}

时间复杂度为O(logn)

空间复杂度为O(1) 

算法示例 

 以nums = [4,5,6,7,0,1,2]为例

 第一步:初始化

left=0,right=6

第二步:预处理

此时nums[left]=4>nums[right]=7,继续进行下续操作

第三步:找最小值

  1. mid=left+(right-left)/2=0+(6-0)/2=3,nums[mid]=7>nums[right]=2,此时让left=mid+1=4

2.mid=4+(6-4)/2=5,nums[mid]=1<nums[right]=2,让right=mid=5

 

3.mid=4+(5-1)/2=4,nums[mid]=0<nums[right]=1,此时让right=mid,同时left和right相遇,结束循环。

第四步:返回结果

此时返回nums[left]=0即可。

点名 

 

算法分析

本道题是要在一个从0~n-1的数组中找缺失的数,我们可以采用哈希表来解决,但此时时空复杂度达到了O(n),这是一个有序的数组,我们可以采用二分查找来解决,使时间复杂度达到O(logn).我们可以发现,每个数组的下标和其元素是相同的,那么我们可以通过判断下标和元素的大小,来确定缺失值的位置。

算法步骤

  1. 初始化:设置双指针left和right,初始化left=0,right=nums.length-1
  2. 查找缺失数:定义mid,mid=left+(right-left)/2,当records[mid]==mid,说明此时在[left,mid]内的数是没有缺失的,让left=mid+1,当records[mid]!=mid,说明此时缺失值在[left,mid]中。循环条件为:left<right(当left和right相遇,说明可能找到了缺失值的位置)
  3. 返回结果:在返回left下标前,我们需要判断一下records[left]==left,若是相等,说明此时不缺值,返回left+1,反之,返回left即可。

算法代码

/*** 记录出席情况的函数。* 通过数组记录每个人的出席情况,其中数组的索引代表人的编号,数组的值代表该人实际的出席编号。* 函数的目的是找到第一个未出席的人的编号。** @param records 出席记录数组,数组的第i个元素表示第i个人的出席编号。* @return 返回第一个未出席的人的编号。如果所有人都出席了,则返回下一个应该出席的编号。*/
public int takeAttendance(int[] records) {/* 初始化左右指针 */int left = 0, right = records.length - 1;/* 使用二分查找法来寻找第一个未出席的人 */while (left < right) {/* 计算中间位置,避免整数溢出 */int mid = left + (right - left) / 2;/* 如果中间位置的人的出席编号等于其位置,则说明左边的人都出席了,调整左指针 */if (records[mid] == mid) {left = mid + 1;} else {/* 否则,说明中间位置的人未出席或者出席编号在左边,调整右指针 */right = mid;}}/* 检查最后一个人是否出席,如果出席则返回下一个出席编号,否则返回当前出席的最后一个编号 */return left == records[left] ? left + 1 : left;
}

时间复杂度为O(logn),

空间复杂度为O(1) 

算法示例 

records = [0,1,2,3,5]

第一步:初始化

left=0,right=4

第二步:查找缺失值

  1. mid=left+(right-left)/2=0+(4-0)/2=2,records[mid]=2=mid,让left=mid+1

2.mid=3+(4-3)/2=3,records[mid]=3=mid,让left=mid+1,此时left=right,结束循环。

第三步:返回结果

此时判断records[left]是否等于left,但通过判断不是,所以这里直接返回left=4即可。

 


二分查找的算法专题就先到这里了~

若有不足,欢迎指正~

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

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

相关文章

TCP/IP协议(全的一b)应用层,数据链层,传输层,网络层,以及面试题

目录 TCP/IP协议介绍 协议是什么,有什么作用? 网络协议为什么要分层 TCP/IP五层网络协议每层的作用 应⽤层 DNS的作用及原理 DNS工作流程 数据链路层 以太⽹帧格式 MAC地址的作用 ARP协议的作⽤ ARP协议的工作流程 MTU以及MTU对 IP / UD / TCP 协议的影响 传输层…

数据结构之判断二叉树是否为搜索树(C/C++实现)

文章目录 判断二叉树是否为搜索树方法一&#xff1a;递归法方法二&#xff1a;中序遍历法总结 二叉树是一种非常常见的数据结构&#xff0c;它在计算机科学中有着广泛的应用。二叉搜索树&#xff08;Binary Search Tree&#xff0c;简称BST&#xff09;是二叉树的一种特殊形式&…

自动化测试--WebDriver API

1. 元素定位方法 通过 ID 定位&#xff1a;如果元素具有唯一的 ID 属性&#xff0c;可以使用 findElement(By.id("elementId")) 方法来定位元素。通过 Name 定位&#xff1a;使用 findElement(By.name("elementName")) 来查找具有指定名称的元素。通过 Cl…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] 5键键盘(100分) - 三语言AC题解(Python/Java/Cpp)

🍭 大家好这里是清隆学长 ,一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 💻 ACM银牌🥈| 多次AK大厂笔试 | 编程一对一辅导 👏 感谢大家的订阅➕ 和 喜欢💗 🍿 最新华为OD机试D卷目录,全、新、准,题目覆盖率达 95% 以上,支持题目在线…

Linux常用操作

软件安装 CentOS 系统使用&#xff1a; yum [install remove search] [-y] 软件名称 install 安装 remove 卸载 search 搜索 -y &#xff0c;自动确认 Ubuntu 系统使用 apt [install remove search] [-y] 软件名称 install 安装 remove 卸载 search 搜索 -y…

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第四十六章 自动创建设备节点

i.MX8MM处理器采用了先进的14LPCFinFET工艺&#xff0c;提供更快的速度和更高的电源效率;四核Cortex-A53&#xff0c;单核Cortex-M4&#xff0c;多达五个内核 &#xff0c;主频高达1.8GHz&#xff0c;2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…

前端播放rtsp视频流(最后使用WebRtc)

前端播放rtsp视频流&#xff08;最后使用WebRtc&#xff09; 前言&#xff1a; ​ 项目需要将实验室里的摄像头画面引入到前端页面中&#xff0c;故对目前常见的几种方法进行了尝试&#xff0c;虽然过程坎坷但结局是好的。 一些尝试&#xff1a; RTSPtoWebRtc工具 由于RTSPt…

opengl 写一个3D立方体——计算机图形学编程 第4章 管理3D图形数据 笔记

计算机图形学编程&#xff08;使用OpenGL和C&#xff09; 第4章 管理3D图形数据 笔记 数据处理 想要绘制一个对象&#xff0c;它的顶点数据需要发送给顶点着色器。通常会把顶点数据在C端放入 一个缓冲区&#xff0c;并把这个缓冲区和着色器中声明的顶点属性相关联。 初始化立…

力扣 二分查找

二分查找基础篇。 题目 class Solution {public int searchInsert(int[] nums, int target) {int l 0, r nums.length - 1;while(l < r) {int mid l((r-l)>>1);//(lr)/2if(nums[mid]<target)lmid1;else rmid-1;}return l;//处理边界&#xff0c;设定数组的左半…

21 Python常用内置函数——zip()

zip() 函数用来把多个可迭代对象中的元素压缩到一起&#xff0c;返回一个可迭代的 zip 对象&#xff0c;其中每个元素都是包含原来的多个可迭代对象对应位置上元素的元组&#xff0c;最终结果中包含的元素个数取决于所有参数序列或可迭代对象中最短的那个。 可以这样理解这个函…

论文阅读——Design of Environmental backscatter tag antenna for 5G Internet of things

文章目录 摘要一、背景二、系统模型三、天线设计A. 指标B. 天线结构描述C. 天线结构优化D. 天线结构确定 四、仿真结果总结 论文来源&#xff1a;https://ieeexplore.ieee.org/document/9379395 摘要 文章针对传统设备识别在电力物联网场景中存在的可靠性低和读取距离不足的问…

Java智慧养老养老护理帮忙代办陪诊陪护小程序系统源码

&#x1f31f;智慧养老新风尚&#xff0c;护理代办陪诊小程序来帮忙✨ &#x1f3e1;【开篇&#xff1a;关爱老人&#xff0c;从智慧养老开始】&#x1f3e1; 随着社会的进步&#xff0c;智慧养老已成为新时代孝心的体现。面对忙碌的生活节奏&#xff0c;如何更好地照顾家中长…

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第五十九章 等待队列

i.MX8MM处理器采用了先进的14LPCFinFET工艺&#xff0c;提供更快的速度和更高的电源效率;四核Cortex-A53&#xff0c;单核Cortex-M4&#xff0c;多达五个内核 &#xff0c;主频高达1.8GHz&#xff0c;2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…

电力系统 | 发电、输电、变电、配电、用电介绍 | 一度电从电厂发出来到用户终端需要经历哪些环节 | 变电站建在哪里

文章目录 一、一度电从电厂发出来到用户终端需要经历哪些环节&#xff1f;二、发电、变电、输电、配售电和用电过程介绍三、变电站建在哪里&#xff1f; 一、一度电从电厂发出来到用户终端需要经历哪些环节&#xff1f; 电力系统是由发电、变电、输电、配售电和用电等环节组成的…

leetcode106. 从中序与后序遍历序列构造二叉树,力扣105姊妹题

leetcode106. 从中序与后序遍历序列构造二叉树 给定两个整数数组 inorder 和 postorder &#xff0c;其中 inorder 是二叉树的中序遍历&#xff0c; postorder 是同一棵树的后序遍历&#xff0c;请你构造并返回这颗 二叉树 。 示例 1: 输入&#xff1a;inorder [9,3,15,20,7…

活动报名小程序

#活动报名工具# # 活动报名小程序 ## 项目简介 一款通用的活动报名工具&#xff0c;包含活动展示&#xff0c;微信支付&#xff0c;订单管理&#xff0c;分享评价等功能。 品客聚精彩&#xff0c;有你才精彩&#xff01;不只有线下活动还可以进行线上裂变活动。 …

UE4-构建光照后导入的静态网格体变黑

当我们将我们的静态网格体导入到项目当中的时候&#xff0c;此时我们进行重新构建光照&#xff0c;我们在从新构建完光照后&#xff0c;会发现我们的静态网格体全部变黑了&#xff0c;此时是因为没有设置光照贴图分辨率和坐标索引引起的。 将General Settings中的L…

Cmake生成的Xcode工程相对路径与绝对路径的问题

Cmake生成的Xcode工程相对路径与绝对路径的问题 文章目录 Cmake生成的Xcode工程相对路径与绝对路径的问题前言修改.pbxproj文件验证工程小结 前言 由于Cmake的跨平台的自动化构建的方便性以及他广泛应用于编译过程的管理&#xff0c;在开发过程中难免用到Cmake。我也使用Cmake…

framework直播学习笔记--安卓如何实现Launcher启动应用全部变自由窗口Freeform模式

背景&#xff1a; 前些天在学员在学员群里有聊到一个需求&#xff0c;那就是把手机桌面点击应用图标后&#xff0c;不是进行全屏显示&#xff0c;而是都进行自由窗口显示。这个其实有点类似我们windows电脑打开app&#xff0c;每个app都是一个非全屏的窗口&#xff0c;而且可以…

从 Batch Norm 到 SGD 隐藏的内容

我们仍然不了解机器学习的哪些方面 欢迎来到雲闪世界。令人惊讶的是&#xff0c;机器学习中的一些基本主题仍然不为研究人员所知&#xff0c;尽管它们很基础且常用&#xff0c;但却似乎很神秘。机器学习的有趣之处在于我们构建了可以工作的东西&#xff0c;然后弄清楚它们为什么…