【算法系列-数组】二分查找

news2024/9/27 5:49:34

【算法系列-数组】二分查找

文章目录

  • 【算法系列-数组】二分查找
    • 1. 算法分析
    • 2. 搜索插入位置(LeetCode 35)
      • 2.1 解题思路
      • 2.2 解题过程
      • 2.3 代码
    • 3. 在排序数组中查找元素的第一个和最后一个位置(LeetCode 34)
      • 3.1 解题思路
      • 3.2 解题过程
      • 3.3 代码
    • 4. X的平方根(LeetCode 69)
      • 4.1 解题思路
      • 4.2 代码
    • 5. 有效的完全平方数(LeetCode 367)
      • 5.1 解题思路
      • 5.2 代码

1. 算法分析

二分查找这类算法题大家并不陌生,对于它的使用方式主要有两个难点,也是最容易出错的地方,就是分不清边界值

LeetCode:二分查找

一般二分查找分为两个场景:

  • 左闭右闭区间,如:提供数组[1, 5]
  • 左闭右开区间,如:提供数组[1, 5)

对此,只需要分清楚在什么场景下使用合适的边界值就能很好的写出这类题,接下来帖上两段代码:

  1. 左闭右闭

    class Solution {
        public int search(int[] nums, int target) {
            int left = 0;
            int right = nums.length -1;
            while (left <= right) { // 这里使用 `<=` ,因为在左闭右闭区间中,left = right 是合理逻辑,对此需要加上等号
                int mid = left + (right - left) / 2;
                if (nums[mid] > target) {
                    right = mid -1; // 因为此处 nums[mid] 不等于目标值,不需要取到mid,所以在左闭右闭区间 right = mid - 1
                }
                else if (nums[mid] < target) {
                    left = mid + 1;
                }
                else {
                    return mid;
                }
            }
            return -1;
        }
    }
    
  2. 左闭右开

    class Solution {
        public int search(int[] nums, int target) {
            int left = 0;
            int right = nums.length -1;
            while (left < right) { // 这里使用 `<` ,因为在左闭右开区间中,left = right 是不合理逻辑,对此无需加上等号
                int mid = left + (right - left) / 2;
                if (nums[mid] > target) {
                    right = mid; // 因为此处 nums[mid] 不等于目标值,不需要取到mid,而右开区间不取右值,所以right = mid,假设使用 mid - 1,则此时 mid - 1 的值将不会被取到,从而产生问题
                }
                else if (nums[mid] < target) {
                    left = mid + 1;
                }
                else {
                    return mid;
                }
            }
            return -1;
        }
    }
    

掌握解题思路后,以下几道相同类型的题可用来练习:

2. 搜索插入位置(LeetCode 35)

【题目链接】

2.1 解题思路

二分法能够解决这个问题,最主要的是在值不存在的情况下二分法得出的mid让我们节省来再去循环遍历的过程

2.2 解题过程

二分查找流程

先定义左右下标,将mid放在循环外面创建,因为当前数组为闭区间(左闭右闭),所以while循环使用left <= right 合理(如[1,1]),之后,当中间值大于目标值时,右下标 = mid - 1;当中间值小于目标值时,right = mid + 1;当中间值等于目标值时,返回mid;

查无目标值

当循环结束还没有找到目标值时,只有三种情况:

  1. right小于0,表示目标值小于数组中的所有值,返回left(此时left = 0);
  2. left大于数组长度,表示目标值大于数组中的所有值,返回left(此时left = num.length);
  3. 以上两者都不符合表示目标值需在数组中插入,此时right一定在left的左边才能退出循环,返回left(即left的位置即是插入位置)

综上所述,查无目标值的情况返回left即可

2.3 代码

class Solution {
    public int searchInsert(int[] nums, int target) {
        int left = 0;
        int right = nums.length - 1;
        int mid = -1;
        while (left <= right) {
            mid = left + (right - left) / 2;
            if (nums[mid] > target) {
                right = mid - 1;
            }
            else if (nums[mid] < target){
                left = mid + 1;
            }
            else {
                return mid;
            }
        }
        return left;
    }
}

3. 在排序数组中查找元素的第一个和最后一个位置(LeetCode 34)

【题目链接】

3.1 解题思路

二分查找+递归实现

3.2 解题过程

这道题的难点在于数组中可能存在多个相同的目标值,我们通过二分查找找到一个目标值后还需要进行判断,找到第一个和最后一个相同的目标值的下标,对此可以通过递归的方式来解决:

在找到一个目标值后,依次向左向右查询相同值

  • 向左递归:传入数组与当前目标值下标mid,判断 mid - 1 位置的值是否与当前目标值相同,若相同则传入mid - 1继续递归,否则返回当前目标值的下标**mid,**此时的mid即目标值在数组中的第一个位置;
  • 向右递归:传入数组与当前目标值下标mid,判断 mid + 1 位置的值是否与当前目标值相同,若相同则传入mid + 1继续递归,否则返回当前目标值的下标**mid,**此时的mid即目标值在数组中的最后一个位置;

最后返回结果赋值即可

3.3 代码

class Solution {
    public int[] searchRange(int[] nums, int target) {
        int left = 0;
        int right = nums.length - 1;
        int ret[] = new int[2];
        ret[0] = -1;
        ret[1] = -1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (nums[mid] > target) {
                right = mid - 1;
            }
            else if (nums[mid] < target) {
                left = mid + 1;
            } 
            else {
                ret[0] = searchLeft(nums, mid);
                ret[1] = searchRight(nums, mid);
                return ret;
            }
        }
        return ret;
    }

    public int searchLeft(int[] nums, int mid) {
        
        if (mid - 1 >= 0 && nums[mid - 1] == nums[mid]) {
            mid = searchLeft(nums,  mid - 1);
        }
        return mid;
    }

    public int searchRight(int[] nums, int mid) {
        
        if (mid + 1 < nums.length && nums[mid + 1] == nums[mid]) {
            mid = searchRight(nums,  mid + 1);
        }
        return mid;
    }
}

4. X的平方根(LeetCode 69)

【题目链接】

4.1 解题思路

因为这道题限制了我们使用api,所以可以通过二分查找的方式来解决

4.2 代码

class Solution {
    public int mySqrt(int x) {
        int left = 0;
        int right = x;
        int ret = -1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if ((long) mid * mid <= x) {
                ret = mid;
                left = mid + 1;
            }
            else {
                right = mid - 1;
            }
        }
        return ret;
    }
}

5. 有效的完全平方数(LeetCode 367)

【题目链接】

5.1 解题思路

与上道题很相似,只是返回的结果不同了,稍微调整即可

5.2 代码

class Solution {
    public boolean isPerfectSquare(int num) {
        int left = 0;
        int right = num;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if ((long) mid * mid < num) {
                left = mid + 1;
            }
            else if ((long) mid * mid > num) {
                right = mid - 1;
            }
            else {
                return true;
            }
        }
        return false;
    }
}

以上便是对二分查找的介绍了!!后续还会继续分享其它算法系列内容,如果这些内容对大家有帮助的话请给一个三连关注吧💕( •̀ ω •́ )✧( •̀ ω •́ )✧✨

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

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

相关文章

【中级通信工程师】终端与业务(四):通信产品

【零基础3天通关中级通信工程师】 终端与业务(四)&#xff1a;通信产品 本文是中级通信工程师考试《终端与业务》科目第四章《通信产品》的复习资料和真题汇总。终端与业务是通信考试里最简单的科目&#xff0c;有效复习通过率可达90%以上&#xff0c;本文结合了高频考点和近几…

计算机毕业设计PySpark+Django深度学习游戏推荐系统 游戏可视化 游戏数据分析 游戏爬虫 Scrapy 机器学习 人工智能 大数据毕设

本论文的主要研究内容如下&#xff1a; 了解基于Spark的TapTap游戏数据分析系统的基本架构&#xff0c;掌握系统的开发方法&#xff0c;包括系统开发基本流程、开发环境的搭建、测试与运行等。 主要功能如下&#xff1a; &#xff08;1&#xff09;用户管理模块&#xff1a;…

OpenCV图像文件读写(5)从文件系统中读取图像的标准函数imread()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 从文件加载图像。 imread 函数从指定的文件加载图像并返回它。如果图像无法读取&#xff08;因为文件缺失、权限不当、格式不受支持或无效&…

IDEA使用技巧和插件推荐

作为专业开发人员&#xff0c;选择一个强大的集成开发环境&#xff08;IDE&#xff09;是至关重要的。IntelliJ IDEA&#xff08;简称IDEA&#xff09;不仅功能丰富&#xff0c;而且操作便捷&#xff0c;是众多开发者的首选。本文将分享一些使用IDEA的技巧&#xff0c;以及推荐…

测试部署单副本 oceanbase-3.2.4.1 企业版

由于项目需要&#xff0c;测试部署单副本 oceanbase-3.2.4.1 企业版 1.安装前提 准备4cpu,12G内存,100G磁盘 统为centos7.9 yum install -y yum-utils wget net-tools tree yum-config-manager --add-repo https://mirrors.aliyun.com/oceanbase/OceanBase.repo 2.创建用…

SLF4J报错log4j又报错

项目场景&#xff1a; 搭建一个spirngboot项目&#xff0c;启动运行时&#xff0c;SLF4J报错 解决后 ~ log4j又报错了。 问题描述 首先是SLF4J报错了&#xff0c;解决完SL4J报错问题后&#xff0c;再次启动项目&#xff0c;log4j又报错了 。。。 报错信息&#xff1a; SLF4J…

IDEA Dependency Analyzer 分析 maven 项目包的依赖

一、场景分析 javax.validation 是我们 SpringMVC 常用的数据校验框架。但是 javax.validation 是一个规范&#xff08;Java Bean Validation&#xff0c;简称 JSR 380&#xff09;&#xff0c;它并没有具体的实现&#xff0c;它的常用实现&#xff0c;是hibernate-validator。…

【知了社保-注册安全分析报告-无验证方式导致安全隐患】

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 1. 暴力破解密码&#xff0c;造成用户信息泄露 2. 短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉 3. 带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造…

CentOs-Stream-9 解决此系统未向授权服务器注册问题

RPM包注册安装问题 无法执行yum安装rpm问题(后面安装Mysql会遇到)&#xff1a;This system is not registered with an entitlement server. You can use "rhc" or "subscription-manager" to register. Mysql5.7本地无法注册&#xff1a; Mysql8本地无法…

EtherNET IP 转 Profinet 网关:工业通信的桥梁

在工业自动化领域&#xff0c;不同设备之间的通信协议兼容性是一个关键问题。EtherNET IP 和 Profinet 作为两种常用的工业以太网协议&#xff0c;各自在不同的设备和系统中广泛应用。而 EtherNET IP 转 Profinet 网关则成为了连接这两个不同协议世界的桥梁&#xff0c;发挥着至…

OJ在线评测系统 后端 用策略模式优化判题机架构

判题机架构优化(策略模式) 思考 我们的判题策略可能会有很多种 比如 我们的代码沙箱本身执行程序需要消耗时间 这个时间可能不同的编程语言是不同的 比如沙箱执行Java要额外花费2秒 我们可以采用策略模式 针对不同的情况 定义不同独立的策略 而不是把所有情况全部放在一个i…

【Mysql】数据库系统和Mysql

1、数据库系统 数据库&#xff08;Database&#xff09;是一个以某种组织方式存储在磁盘上的数据当代集合。 2、数据库应用 数据库应用系统是指基于数据库的应用软件。 3、数据库管理系统&#xff08;数据库软件&#xff09; &#xff08;1&#xff09;关系型数据库&#…

【JAVA高级】 redis分布式双重加锁(业务校验:防止接口并发调用时数据重复)

文章目录 此问题的考虑思路使用Redis的key-value锁的基本思路结合Redis数据结构实现避免重复注意事项实现代码只避免 name和age的重复避免 name和age的和age和sex重复&#xff1a;使用双重的分布式锁实现&#xff1a; 背景&#xff1a;在日常开发过程中&#xff0c;遇到了一个需…

FGPA实验——触摸按键

本文系列都基于正点原子新起点开发板 FPGA系列 1&#xff0c;verlog基本语法&#xff08;随时更新&#xff09; 2&#xff0c;流水灯&#xff08;待定&#xff09; 3&#xff0c;FGPA实验——触摸按键 一、触摸操作原理实现 分类&#xff1a;电阻式&#xff08;不耐用&…

SVN文件不显示修改状态图标

今天安装试用SVN时发现文件不显示修改状态 以下为解决方法&#xff1a; 1&#xff0c;在有.svn的文件夹中右键--tortoiseSvn--setting 2&#xff0c;选中icon Overlays&#xff0c;右侧的status cache 选shell 3&#xff0c;点击icon set 如下图所示 4&#xff0c;修改icon…

MySQL扩展

一、慢查询&#xff08;慢日志&#xff09; 默认关闭的 定位慢SQL 简单&#xff1a;show profile&#xff0c;启用时会对服务器的性能产生额外的负担 -- 启用性能监控 mysql> set profiling1;-- 执行SQL mysql> SELECT * from member-- 性能分析 mysql> show p…

AOT源码解析4.4 -decoder生成预测mask并计算loss

3、生成ref_imgs的预测mask和loss 这一步在训练阶段调用 3.1 数据处理 图1&#xff0c;如图1所示&#xff0c;将enc_embs的最后一个比例的特征图和有ref_imgs相关的特征图得到的LSTT特征图相拼接作为输入 curr_enc_embs self.curr_enc_embscurr_lstt_embs self.curr_lstt_o…

卷轴模式商城APP开发搭建全流程解析

卷轴模式商城APP的开发搭建是一个综合性强、涉及多个关键步骤和技术环节的过程。本文将详细介绍从需求分析到最终发布的各个阶段&#xff0c;旨在为开发者renxb001提供一个清晰的开发指导方案。 一、需求分析 目标用户群体&#xff1a;首先&#xff0c;明确APP的目标用户&…

openKylin--安装 .net6.0

编辑profile文件 cd .. //切换到根目录 cd /etc //切换到etc目录 vim profile //b编辑profile文件 1. 按→键移动到文件末尾 2. 按Insert键进入编辑模式 3. 按Enter另起一行开始编辑 export DOTNET_ROOT/home/dotnetexport PATH$PATH:/home/dotnet 可以通过右键--粘贴 的…

基于skopt的贝叶斯优化基础实例学习实践

贝叶斯方法是非常基础且重要的方法&#xff0c;在前文中断断续续也有所介绍&#xff0c;感兴趣的话可以自行移步阅读即可&#xff1a; 《数学之美番外篇&#xff1a;平凡而又神奇的贝叶斯方法》 《贝叶斯深度学习——基于PyMC3的变分推理》 《模型优化调参利器贝叶斯优化bay…