【无标题】

二分算法

核心原理

二分算法(Binary Search),又称折半查找,是一种高效的查找算法,用于在有序集合中快速定位目标元素。其核心思想是"每次通过与中间元素比较,将查找范围缩小一半",从而实现对数级的时间复杂度

核心逻辑

  1. 确定边界:定义查找范围的左边界(left)和右边界(right),初始时左边界为集合起始索引(通常是0),右边界为集合末尾索引(通常是length-1)。

  2. 计算中点:计算当前范围的中间索引mid = left + (right - left) // 2(用这种方式而非(left+right)//2,是为了避免left和right过大时出现整数溢出)。

  3. 比较判断

    若目标值 == 中间元素(nums[mid]):找到目标,返回mid。

  4. 若目标值 < 中间元素:目标在左半部分,将右边界更新为mid - 1。

  5. 若目标值 > 中间元素:目标在右半部分,将左边界更新为mid + 1。

  6. 循环终止:重复步骤2-3,直到左边界 > 右边界,此时说明集合中无目标元素,返回-1或其他标识。

关键特性

  • 时间复杂度:O(log n),n为集合元素个数,每轮查找范围减半,效率远高于线性查找(O(n))。

  • 空间复杂度:递归实现为O(log n)(栈空间),迭代实现为O(1)(仅用几个变量)。

  • 适用前提:必须是有序集合(升序或降序,需统一判断逻辑),且集合支持随机访问(如数组,链表不适用,因无法快速定位mid)

实战

问题

给定升序数组nums和目标值target,若数组中有多个target,返回第一个出现的索引;若不存在,返回-1

思路

当nums[mid] == target时,不直接返回,而是将右边界收缩到mid(而非mid-1),继续向左查找,直到左边界>右边界,最终判断左边界是否为目标值。

实现

java 复制代码
import java.util.Arrays;

public class FindLeftBoundDemo {
    // 查找目标值的左边界(第一个等于target的元素)
    public static int findLeftBound(int[] nums, int target) {
        int left = 0;
        int right = nums.length - 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 {
                right = mid - 1;
            }
        }
        // 循环结束后,left可能是左边界,也可能不存在
        if (left < nums.length && nums[left] == target) {
            return left;
        }
        return -1;
    }

    // 测试案例
    public static void main(String[] args) {
        int[] nums = {1, 2, 2, 2, 3, 4};
        int target = 2;
        System.out.println(findLeftBound(nums, target));  // 输出:1(第一个2的索引)

        int target2 = 5;
        System.out.println(findLeftBound(nums, target2));  // 输出:-1
    }
}

注意事项

  1. 边界条件处理

    循环条件用left <= right还是left <?取决于是否要"包含边界元素"。若用left <,右边界初始需设为len(nums)(而非len(nums)-1),适用于"找插入位置"等场景。

  2. 避免数组越界:判断结果时需检查left是否在数组范围内(如左边界查找时的left < len(nums))。

  3. 中点计算 :必须用mid = left + (right - left) // 2,而非(left + right) // 2。例如left=231-1、right=231-1时,left+right会超出Python整数范围(虽Python支持大数,但其他语言如Java会溢出)。

  4. 有序性保证:若数组无序,必须先排序(但排序会消耗O(n log n)时间,若仅查找一次,不如直接线性查找;若多次查找,排序+二分更高效)。

  5. 重复元素处理:普通二分可能返回任意一个重复元素的索引,需通过"边界查找"逻辑定位第一个或最后一个。

相关推荐
金色旭光1 小时前
目标追踪算法+卡尔曼滤波原理+ByteTrack使用
算法
Knight_AL2 小时前
如何在 Spring Boot 中集成 IP2Region 实现高效 IP 地址地理位置查询
java·spring boot·tcp/ip
山枕檀痕2 小时前
Spring Boot中LocalDateTime接收“yyyy-MM-dd HH:mm:ss“格式参数的最佳实践
java·spring boot·后端
乔伊酱2 小时前
Bean Searcher 遇“鬼”记:为何我的查询条件偷偷跑进了 HAVING?
java·前端·orm
invicinble2 小时前
idea提供maven处理机制
java·maven·intellij-idea
fantasy5_52 小时前
C++11 核心特性实战博客
java·开发语言·c++
喜欢流萤吖~2 小时前
Java函数式接口详解
java
夏乌_Wx2 小时前
练题100天——DAY22:数字拼接+只出现一次的数字
java·数据结构·算法
listhi5202 小时前
MOEAD算法实现详解(基于Python与MATLAB)
python·算法·matlab