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

数的范围题解(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
相关推荐
薛定谔的猫喵喵几秒前
基于PyQt5的视频答题竞赛系统设计与实现
开发语言·qt·音视频
岱宗夫up6 分钟前
Python 数据分析入门
开发语言·python·数据分析
xhbaitxl6 分钟前
算法学习day38-动态规划
学习·算法·动态规划
多恩Stone7 分钟前
【3D AICG 系列-6】OmniPart 训练流程梳理
人工智能·pytorch·算法·3d·aigc
wangjialelele8 分钟前
Linux中的进程管理
java·linux·服务器·c语言·c++·个人开发
码界筑梦坊8 分钟前
325-基于Python的校园卡消费行为数据可视化分析系统
开发语言·python·信息可视化·django·毕业设计
历程里程碑9 分钟前
普通数组----轮转数组
java·数据结构·c++·算法·spring·leetcode·eclipse
pp起床10 分钟前
贪心算法 | part02
算法·leetcode·贪心算法
sin_hielo10 分钟前
leetcode 1653
数据结构·算法·leetcode
多恩Stone11 分钟前
【RoPE】Flux 中的 Image Tokenization
开发语言·人工智能·python