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

数的范围题解(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
相关推荐
消失的旧时光-19432 分钟前
第十三课:权限系统如何设计?——RBAC 与 Spring Security 架构
java·架构·spring security·rbac
baidu_247438613 分钟前
Android ViewModel定时任务
android·开发语言·javascript
Dev7z16 分钟前
基于 MATLAB 的铣削切削力建模与仿真
开发语言·matlab
不能隔夜的咖喱22 分钟前
牛客网刷题(2)
java·开发语言·算法
VT.馒头22 分钟前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript
serve the people28 分钟前
python环境搭建 (十二) pydantic和pydantic-settings类型验证与解析
java·网络·python
小天源29 分钟前
Error 1053 Error 1067 服务“启动后立即停止” Java / Python 程序无法后台运行 windows nssm注册器下载与报错处理
开发语言·windows·python·nssm·error 1053·error 1067
进击的小头39 分钟前
实战案例:51单片机低功耗场景下的简易滤波实现
c语言·单片机·算法·51单片机
lekami_兰43 分钟前
Java 并发工具类详解:4 大核心工具 + 实战场景,告别 synchronized
java·并发工具
有位神秘人1 小时前
Android中Notification的使用详解
android·java·javascript