首页 > 编程语言 >搜索算法之二分搜索详细解读(附带Java代码解读)

搜索算法之二分搜索详细解读(附带Java代码解读)

时间:2024-09-06 16:23:50浏览次数:7  
标签:arr right Java mid 搜索算法 解读 搜索 目标值 left

1. 基本概念

二分搜索(Binary Search)是一种高效的查找算法,用于在一个已排序的数组中查找特定元素。它通过逐步将搜索范围减少一半来实现搜索,从而比线性搜索更快。由于它利用了数组的有序性,能够在对数时间内完成搜索操作。

2. 工作原理

二分搜索的基本思想是:

  1. 初始化:设置两个指针,leftright,分别指向数组的开始和结束位置。

  2. 中间点:计算当前搜索范围的中间点 mid,即 (left + right) / 2

  3. 比较

    • 如果 arr[mid] 等于目标值 target,则找到目标值,返回 mid
    • 如果 arr[mid] 大于目标值 target,则目标值在 leftmid - 1 之间,调整 rightmid - 1
    • 如果 arr[mid] 小于目标值 target,则目标值在 mid + 1right 之间,调整 leftmid + 1
  4. 结束条件:重复步骤 2 和 3,直到 left 超过 right。如果在这个过程中没有找到目标值,返回 -1,表示目标值不在数组中。

3. 算法步骤

以下是二分搜索的详细步骤:

  1. 输入

    • 一个已排序的整数数组 arr
    • 一个目标值 target
  2. 步骤

    • 设置 left 为 0。
    • 设置 rightarr.length - 1
    • 进入循环,直到 left 小于或等于 right
      • 计算中间点 midmid = (left + right) / 2
      • 如果 arr[mid] 等于 target,返回 mid
      • 如果 arr[mid] 大于 target,将 right 设置为 mid - 1
      • 如果 arr[mid] 小于 target,将 left 设置为 mid + 1
    • 如果循环结束后仍未找到目标值,返回 -1
4. 时间复杂度分析
  • 最坏情况:每次比较后,搜索范围缩小一半。对于长度为 n 的数组,最多需要 log2(n) + 1 次比较,时间复杂度为 O(log n)。

  • 最佳情况:目标值恰好是中间元素,时间复杂度为 O(1)。

  • 平均情况:时间复杂度为 O(log n),因为每次搜索范围减少一半。

5. 空间复杂度
  • 空间复杂度:二分搜索只需要常数级别的额外空间来存储变量 leftrightmid,因此空间复杂度为 O(1)。
6. 实现代码
public class BinarySearch {

    /**
     * 执行二分搜索
     * @param arr 已排序的数组
     * @param target 目标值
     * @return 目标值的索引,如果未找到返回-1
     */
    public static int binarySearch(int[] arr, int target) {
        int left = 0; // 搜索范围的左边界
        int right = arr.length - 1; // 搜索范围的右边界

        while (left <= right) {
            int mid = left + (right - left) / 2; // 计算中间点
            
            if (arr[mid] == target) {
                return mid; // 找到目标值,返回索引
            } else if (arr[mid] < target) {
                left = mid + 1; // 目标值在右半边,调整左边界
            } else {
                right = mid - 1; // 目标值在左半边,调整右边界
            }
        }
        
        return -1; // 未找到目标值
    }

    public static void main(String[] args) {
        // 示例数组(已排序)
        int[] numbers = {1, 2, 4, 7, 9, 10, 15};

        // 目标值
        int target = 7;

        // 执行二分搜索
        int result = binarySearch(numbers, target);

        // 输出搜索结果
        if (result != -1) {
            System.out.println("元素 " + target + " 在数组中的索引是: " + result);
        } else {
            System.out.println("元素 " + target + " 不在数组中。");
        }
    }
}

代码解读

  • public static int binarySearch(int[] arr, int target)

    • 定义了一个静态方法 binarySearch,接受两个参数:一个已排序的整数数组 arr 和一个目标值 target
    • 方法返回目标值的索引,如果未找到则返回 -1
  • int left = 0; int right = arr.length - 1;

    • 初始化 left 为数组的第一个索引,right 为数组的最后一个索引。
  • while (left <= right)

    • 只要 left 小于或等于 right,继续搜索。
  • int mid = left + (right - left) / 2;

    • 计算中间点 mid。使用 left + (right - left) / 2 避免了可能的整数溢出。
  • if (arr[mid] == target)

    • 如果中间点的元素等于目标值,返回 mid
  • else if (arr[mid] < target)

    • 如果中间点的元素小于目标值,将 left 移动到 mid + 1
  • else

    • 如果中间点的元素大于目标值,将 right 移动到 mid - 1
  • return -1;

    • 如果搜索范围为空,返回 -1,表示目标值不在数组中。
7. 实际应用
  • 已排序数据:二分搜索要求数据必须是已排序的。它在有序数组、二叉搜索树等数据结构中非常高效。

  • 查找操作:适用于需要频繁查找操作的场景,如字典查找、数据库索引等。

  • 优化:对于大规模数据集和需要快速查找的情况,二分搜索是非常有效的选择。

8. 变体和改进
  • 递归实现:二分搜索也可以用递归实现,通过将问题分解为子问题来求解。

  • 插值查找:在均匀分布的数据中,插值查找比二分查找更快,通过更智能地选择中间点来减少搜索时间。

  • 扩展应用:如在浮点数数组中的查找、寻找插入位置等。

标签:arr,right,Java,mid,搜索算法,解读,搜索,目标值,left
From: https://blog.csdn.net/m0_61840987/article/details/141962926

相关文章

  • JavaScript 循环语句
    1. for 循环for循环是最常用的循环结构之一,它适合在循环开始前就知道循环次数的情况。基本语法for(初始化表达式;条件表达式;迭代后表达式){//循环体//这里的代码会在每次迭代时执行}如何工作初始化:首先执行初始化表达式,通常用来设置循环控制变量。条件......
  • (免费源码)计算机毕业设计必看必学 原创定制程序 java、PHP、python、小程序、文案全套
    摘 要随着互联网趋势的到来,各行各业都在考虑利用互联网将自己推广出去,最好方式就是建立自己的互联网系统,并对其进行维护和管理。在现实运用中,应用软件的工作规则和开发步骤,采用Java技术建设英语自主学习平台。本设计主要实现集人性化、高效率、便捷等优点于一身的英语自......
  • Java集合框架体系
    Java集合类主要由两个接口Collection和Map派生出来的,Collection有三个子接口:List、Set、Queue。Collection:最基本的集合接口,代表一组元素的集合。List:代表有序的、可重复的元素。Set:代表不可重复的的集合。Queue:代表队列Map:存储键值对的集合,键不允许重复。List、Se......
  • java for循环倒序输出
    在Java中,如果你想使用for循环来实现倒序输出(比如倒序输出一个数组或集合中的元素,或者仅仅是从一个数字倒序输出到另一个数字),有几种方法可以实现。下面是一些常见的示例:示例1:倒序输出数组中的元素假设你有一个整数数组,并希望使用for循环来倒序输出数组中的每个元素。int[]numbers......
  • 数据结构练习题(java版)考前必备!
    今天我们刷一些栈队列的题目,大家还是先看题,后看题解。1.155.最小栈-力扣(LeetCode)思路:创建两个栈,一个栈所有元素都算,另一个栈只放小的元素,第二个栈中如果要放的元素比栈顶的元素小就放,这样我们直接pop第二个栈就能得到最小栈classMinStack{publicStack<Integer>......
  • JAVA三级分类的使用
    1.0准备1.创建好一个java文件2.导入所需要的包(至少29个)3.创建resources包并标记为资源根目录,配置好框架配置信息web.xml4.创建pojo包,编写实体类pojo 5.创建mapper包,编写接口mapper 6.编写实现类mapper.xml  7.创建service包,编写service以及impl8.编写测试......
  • 【Java】【SpringBoot】项目部署
    项目打包SpringBoot项目是依赖于Maven构建的,但打包时如果只依赖Maven打包工具则会打包不完整,我们还需要在SpringBoot项目中引入SpringBoot打包插件: 此时再使用Maven插件打包多环境配置在真实开发中,在不同环境下运行项目往往会进行不同的配置,比如开发环境使用的是开发数据库......
  • 【Java】【SpringBoot】读取配置文件(appliation.yml)的值
    这里叙述4中读取配置文件(application.yml)方法  application.yml配置如下:#测试数据(用于读取数据文件值)student:name:lisiage:13name:zhangsan使用@value注解@SpringBootTestpublicclassApplicationTest{@Value("${student.name}")privateStr......
  • 基于JavaWeb开发的Java+SpringBoot+vue+element实现前后端分离玩具商城系统
    基于JavaWeb开发的Java+SpringBoot+vue+element实现前后端分离玩具商城系统......
  • java webservice 带请求头方式处理
    1、gradle引入依赖的增强第三方包implementation'org.apache.cxf:cxf-spring-boot-starter-jaxws:3.2.6'2、增强类方法packagewebservice;importcom.alibaba.fastjson.JSON;importcom.landray.kmss.sys.notify.webservice.Exception_Exception;importcom.landray.......