二分查找巧解数组范围问题

数的范围题解(Java实现)

问题描述

给定一个升序排列的整数数组 arr 和目标值 k,要求找出 k 在数组中出现的起始位置和结束位置(从0开始计数)。若 k 不存在,则返回 [-1, -1]

算法思路

使用二分查找的变形:

  1. 左边界查找:找到第一个 \\geq k 的位置 $$ \text{while } l < r \text{ 时}, \quad mid = l + r >> 1 $$ $$ \text{若 } arr[mid] \geq k, \text{ 则 } r = mid $$ $$ \text{否则 } l = mid + 1 $$
  2. 右边界查找:找到最后一个 \\leq k 的位置 $$ \text{while } l < r \text{ 时}, \quad mid = l + r + 1 >> 1 $$ $$ \text{若 } arr[mid] \leq k, \text{ 则 } l = mid $$ $$ \text{否则 } r = mid - 1 $$
代码实现
java 复制代码
import java.util.*;

public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt(), q = sc.nextInt();
        int[] arr = new int[n];
        for (int i = 0; i < n; i++) arr[i] = sc.nextInt();
        
        while (q-- > 0) {
            int k = sc.nextInt();
            int l = 0, r = n - 1;
            
            // 左边界查找
            while (l < r) {
                int mid = l + r >> 1; // 向下取整
                if (arr[mid] >= k) r = mid;
                else l = mid + 1;
            }
            
            // 未找到目标值
            if (arr[l] != k) {
                System.out.println("-1 -1");
                continue;
            }
            
            int left = l;
            r = n - 1; // 重置右指针
            
            // 右边界查找
            while (l < r) {
                int mid = l + r + 1 >> 1; // 向上取整
                if (arr[mid] <= k) l = mid;
                else r = mid - 1;
            }
            
            System.out.println(left + " " + l);
        }
    }
}
关键点说明
  1. 左边界查找 :通过 mid = l + r >> 1 确保左指针能推进
  2. 右边界查找 :通过 mid = l + r + 1 >> 1 避免死循环
  3. 边界验证 :左边界查找后需检查 arr[l] == k
  4. 时间复杂度O(\\log n) 每次查询
示例测试

输入:

复制代码
6 3
1 2 2 2 3 4
2
3
4

输出:

复制代码
1 3
4 4
5 5
相关推荐
人工智能AI酱10 分钟前
【AI深究】逻辑回归(Logistic Regression)全网最详细全流程详解与案例(附大量Python代码演示)| 数学原理、案例流程、代码演示及结果解读 | 决策边界、正则化、优缺点及工程建议
人工智能·python·算法·机器学习·ai·逻辑回归·正则化
WangLanguager12 分钟前
逻辑回归(Logistic Regression)的详细介绍及Python代码示例
python·算法·逻辑回归
m0_5180194813 分钟前
C++与机器学习框架
开发语言·c++·算法
ZTLJQ16 分钟前
深入理解逻辑回归:从数学原理到实战应用
开发语言·python·机器学习
一段佳话^cyx18 分钟前
详解逻辑回归(Logistic Regression):原理、推导、实现与实战
大数据·算法·机器学习·逻辑回归
qq_4176950520 分钟前
C++中的代理模式高级应用
开发语言·c++·算法
daidaidaiyu26 分钟前
Spring IOC 源码学习 事务相关的 BeanDefinition 解析过程 (XML)
java·spring
xiaoye-duck44 分钟前
《算法题讲解指南:动态规划算法--路径问题》--5.不同路径,6.不同路径II
c++·算法·动态规划
ambition202421 小时前
最大子数组和算法全解析:从暴力枚举到动态规划优化
数据结构·c++·算法
波波0071 小时前
每日一题:.NET 中的“反射”是什么?
开发语言·.net