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

news2025/2/23 23:44:37

目录

山脉数组的峰顶索引

算法分析 

算法步骤

算法代码

算法示例 

寻找峰值 

算法分析

算法步骤

算法代码

算法示例 

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

算法分析

算法步骤

算法代码

算法示例 

点名 

算法分析

算法步骤

算法代码

算法示例 


山脉数组的峰顶索引

算法分析 

本道题是要在一个先升后降的数组查找拐点,如果我们采用暴力搜索的话,时间复杂度为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.coloradmin.cn/o/1954419.html

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

相关文章

生鲜云订单零售系统小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;用户管理&#xff0c;商品分类管理&#xff0c;商品信息管理&#xff0c;订单评价管理&#xff0c;订单管理&#xff0c;系统管理 微信端账号功能包括&#xff1a;系统首页&#xff0c;商品信息&#…

DDOS攻击学习 - kali初学

文章目录 本地ssh配置nmap(网络连接的工具)nmap -sP IP地址nmap -p 1-65535 -A IP地址主机发现Ping扫描端口扫描时序扫描常用扫描方式指纹识别与探测全端口版本探测防火墙/IDS逃逸报文分段信息收集IP信息收集WHOIS查询数据库渗透测试MySQL列举数据库列举MySQL变量发起请求目录扫…

暑假第一周——网易云音乐仿写

iOS学习 前言发现&#xff1a;控件的堆叠我的&#xff1a;更换头像账号&#xff1a;切换夜间总结 前言 本周正式开始暑期学习&#xff0c;完成了网易云音乐的仿写。 发现&#xff1a;控件的堆叠 发现页面由一个无限轮播图&#xff0c;四个自定义cell完成。无限轮播图在之前已经…

【北京迅为】《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…

css 作业 2

文章目录 前言第四题第五题第六题第七题第八题第九题第十题&#xff08;子标签&#xff09; 前言 昨天写了前面三次作业&#xff0c;今天把剩下的七个作业写完 第四题 http://127.0.0.1:5500/index1.html&#xff0c;就用这个网址查看代码在网页的展示效果 代码评测过不了&…

【C语言】文件操作,文件读写详细介绍

目录 为什么要使用文件&#xff1f; 文件概念 1. 什么是文件&#xff1f; 2. 程序文件 3. 数据文件 4. 文件名 文件的使用 1. 文件指针 2. 文件的打开与关闭 文件的顺序读写 1. 顺序读写函数 2. scanf系列与printf系列 文件的随机读写 1. fseek 2. ftell 3. …

【C++】实验三

题目&#xff1a; 1、如何使用C来找出编码88表示的字符&#xff1f;指出至少两种方法。 思路&#xff1a;方法一定义一个字符型变量直接等于88&#xff0c;将其输出结果为编码88表示的字符&#xff1b;方法二使用整形变量来存储88&#xff0c;输出时将其强制转换成字符型。 …

GD32 MCU如何将烧录口配置为GPIO使用?

如果大家在进行GD32 MCU开发时发现GPIO引脚使用不足&#xff0c;可以尝试将烧录口配置为GPIO使用&#xff0c;这样就可以多出几个引脚使用&#xff0c;但使用的时候如何配置以及有哪些注意事项&#xff0c;本视频将会为大家进行解答。 GD32 MCU存在两种GPIO备用功能的配置&…

数字陷波器的设计和仿真(Matlab+C)

目录 一、数字陷波器的模型 二、Matlab仿真 1. 示例1 2. 示例2 三、C语言仿真 1. 由系统函数计算差分方程 2. 示例代码 一、数字陷波器的模型 二、Matlab仿真 1. 示例1 clear clc f0=100;%滤掉的100Hz fs=1000;%大于两倍的信号最高频率 r=0.9; w0=2*pi*f0/fs;%转换到…

【深度学习】语音合成,TTS,fish-speech

官方项目地址&#xff1a;https://github.com/fishaudio/fish-speech git clone https://github.com/fishaudio/fish-speech.gitdocker run -it --gpus device3 -v /ssd/xiedong/tts:/ssd/xiedong/tts --net host --shm-size 16G kevinchina/deeplearning:pytorch2.3.0-cuda12.…

玩转CSS:用ul li +JS 模拟select,避坑浏览器不兼容。

玩转CSS&#xff1a;用ul li JS 模拟select&#xff0c;避坑浏览器不兼容。 在前端的工作中&#xff0c;经常会遇到 selcet控件&#xff0c;但我们用css来写它的样式时候&#xff0c;总是不那么令人满意&#xff0c;各种浏览器不兼容啊有没有&#xff1f; 那么&#xff0c;我…

Datawhale AI 夏令营——AI+逻辑推理——Task1

# Datawhale AI 夏令营 夏令营手册&#xff1a;从零入门 AI 逻辑推理 比赛&#xff1a;第二届世界科学智能大赛逻辑推理赛道&#xff1a;复杂推理能力评估 代码运行平台&#xff1a;魔搭社区 比赛任务 本次比赛提供基于自然语言的逻辑推理问题&#xff0c;涉及多样的场景&…

27K star!有没有显卡都能搞,快速基于LLM构建本地智能知识库

觉得搞一个AI的智能问答知识库很难吗&#xff1f;那是你没有找对方向和工具&#xff0c; 今天我们分享一个开源项目&#xff0c;帮助你快速构建基于Langchain 和LLM 的本地知识库问答&#xff0c;在GitHub已经获得27K star&#xff0c;它就是&#xff1a;Langchain-Chatchat L…

在Spring项目中使用Maven和BCrypt来实现修改密码功能

简介 在数字时代&#xff0c;信息安全的重要性不言而喻&#xff0c;尤其当涉及到个人隐私和账户安全时。每天&#xff0c;无数的用户登录各种在线服务&#xff0c;从社交媒体到银行账户&#xff0c;再到电子邮件和云存储服务。这些服务的背后&#xff0c;是复杂的系统架构&am…

进程间关系

目录 亲缘关系 进程组关系 会话关系 孤儿态进程 亲缘关系 亲缘关系主要体现于父子进程&#xff0c;子进程父进程创建&#xff0c;代码继承于父进程&#xff0c;父进程负责回收&#xff0c;子进程诞生至结束父进程全程参与&#xff0c;这种称为强亲缘关系。 系统开机后&…

企业级数据分析平台合集介绍

企业发展离不开数据分析&#xff0c;数据分析推动着企业运营、决策和战略规划。它正逐步深入到各行各业的核心业务流程中&#xff0c;从传统的金融、零售、制造业扩展到医疗健康、教育、能源等更多领域。企业正通过数据分析平台实现数据资源的最大化利用&#xff0c;推动业务与…

wireshark--流量分析利器

&#x1f3bc;个人主页&#xff1a;金灰 &#x1f60e;作者简介:一名简单的大一学生;易编橙终身成长社群的嘉宾.✨ 专注网络空间安全服务,期待与您的交流分享~ 感谢您的点赞、关注、评论、收藏、是对我最大的认可和支持&#xff01;❤️ &#x1f34a;易编橙终身成长社群&#…

基于微信小程序的课堂考勤系统的设计与实现(论文+源码)_kaic

基于微信小程序的课堂考勤系统的设计与实现 摘 要 在高校教育普及的今天&#xff0c;学生人数日益增多&#xff0c;为保证课堂质量&#xff0c;教师多要在课前进行考勤。因此本设计提出基于微信小程序的课堂考勤系统&#xff0c;增加了定位功能&#xff0c;避免了“假打卡”…

探索大型语言模型LLama 2:原理揭秘与代码实践

一、引言 1.1 大型语言模型的重要性 大型语言模型作为人工智能领域的重要研究方向&#xff0c;近年来取得了显著的成果。这些模型在自然语言处理、机器翻译、对话系统、文本生成等领域展现了强大的能力&#xff0c;为人类带来了诸多便利。大型语言模型的出现&#xff0c;使得…

卓码软件测评:软件功能测试和非功能测试详情介绍

随着信息技术的不断发展&#xff0c;软件在我们日常生活与工作中扮演着越来越重要的角色。然而&#xff0c;软件质量的好坏直接关系到使用者的体验和企业的声誉。在软件开发过程中&#xff0c;功能测试和非功能测试作为保证软件质量的重要手段&#xff0c;受到了越来越多的关注…