Java【数据结构】二分查找

简介: Java【数据结构】二分查找

🌞 题目:

🌏在有序数组A中,查找目标值target

🌏如果找到返回索引

🌏如果找不到返回-1

image.png

算法实现

 public  int binarySearch(int[] arr,int target) {
        int left = 0;
        int right = arr.length-1;
        while(left<=right) {
            int mid = (left+right)>>>1;
            if(target < arr[mid]) {
                right = mid - 1;
            }
            else if (arr[mid] < target) {
                left = mid + 1;
            }
            else {
                return mid;
            }
        }
        return -1;
    }

注解:

1.为什么while循环条件是left<=right,而不是left<right?

因为当left=right时,mid=left=right可能为我们想要查找的值

2.为什么mid = (left+right)>>>1,而不是(left+right)/2呢? >>>是无符号右移,无符号右移一位相当于除2取整。 不用(left+right)/2原因是,当left+right的值超过int类型的正整数最大范围,其值就会由正变负


在其他的资料中二分查找与这个代码不一样,

✈️ 二分查找的改动版

public static int binarySearch1(int[] arr,int target) {
        int left=0;
        int right = arr.length;      //第一处改动
        while(left < right) {        //第二处改动
            int mid = (left+right)>>>1;
            if(target < arr[mid]) {
                right = mid;          //第三处改动
            }
            else if (arr[mid] < target) {
                left = mid + 1;
            }
            else {
                return mid;
            }
        }
        return -1;
    }

⛵️注解

right=arr.length,作为一个边界存在,left可能为我们的查找目标,但是right一定不是我们要找到的目标

🚀图解演示:

查找13

⛽️在Java中其实已经提供了二分查找的方法binarySearch

public class Test {
    public static void main(String[] args) {
        int[] arr ={1,2,3,4,5,5,6};
        int target = Arrays.binarySearch(arr,3);
        System.out.println(target);
    }
}

🚠运行结果:

2

🚩二分查找对重复元素的处理

📍重复元素最靠右的元素

说明:查找元素为重复元素的话,会查找到最右边的重复元素

Returns:

找到则返回最靠右索引

找不到返回-1

//重复元素最靠右的元素
public class Test5 {
    public static int binarySearch2(int[] arr,int target) {
        int left = 0;
        int right = arr.length-1;
        int cand = -1;
        while (left <= right) {
            int mid = (left + right)>>>1;
            if(target < arr[mid]) {
                right = mid-1;
            } else if (arr[mid] < target) {
                left = mid+1;
            }
            else {
                cand = mid;
               left = mid+1;
            }
        }
        return cand;
    }
}

说明:返回<=target的最右边的索引

Returns:

找到则返回最靠右索引

找不到返回小于target最右边的索引

 public static int binarySearchRightMost(int[] arr,int target){
        int left = 0;
        int right = arr.length-1;
        while(left <= right) {
            int mid = (left + right )>>>1;
            if(target < arr[mid]){
                right = mid-1;
            }
            else {
                left = mid + 1;
            }
        }
        return left-1;
    }

📍重复元素最靠左的元素

说明:查找元素为重复元素的话,会查找到最左边的重复元素

Returns:

找到则返回最靠左索引

找不到返回-1

 public static int binarySearch2(int[] arr,int target) {
        int left = 0;
        int right = arr.length-1;
        int cand = -1;
        while (left <= right) {
            int mid = (left + right)>>>1;
            if(target < arr[mid]) {
                right = mid-1;
            } else if (arr[mid] < target) {
                left = mid+1;
            }
            else {
                cand = mid;
                right = mid - 1;
            }
        }
        return cand;
    }

说明:

返回>=target最左边的索引

Returns:

找到则返回最靠左索引

找不到返回比target大的最左边索引

 public static int binarySearchLeftMost(int[] arr,int target) {
        int left=0;
        int right = arr.length-1;
        while(left <= right) {
            int mid = (left + right)>>>1;
            if(target <= arr[mid]) {
                right = mid - 1;
            }
            else  {
                left = mid + 1;
            }
        }
        return left;
    }

图解:

🚆leetcode二分查找题

1️⃣1.给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1

链接: 二分查找

提示:

你可以假设 nums 中的所有元素是不重复的。

n 将在 [1, 10000]之间。

nums 的每个元素都将在 [-9999,9999]之间。

class Solution {
    public int search(int[] nums, int target) {
        int i=0;
        int j = nums.length-1;
        while(i<=j){
           int m=(i+j)>>>1;
            if(target<nums[m]){
                j=m-1;
            }
            else if(nums[m]<target){
                i=m+1;
            }
            else{
                return m;
            }
        }
        return -1;
    }
}

2️⃣2.给定一个排序的整数数组 nums 和一个整数目标值 target ,请在数组中找到 target ,并返回其下标。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

请必须使用时间复杂度为 O(log n) 的算法。

链接: 搜索插入位置

class Solution {
    public int searchInsert(int[] nums, int target) {
         int left=0;
        int right = nums.length-1;
        while(left <= right) {
            int mid = (left + right)>>>1;
            if(target <= nums[mid]) {
                right = mid - 1;
            }
            else  {
                left = mid + 1;
            }
        }
        return left;
    }
}

3️⃣3.给你一个按照非递减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。

如果数组中不存在目标值 target,返回 [-1, -1]。

你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题.

链接: 在排序数组中查找元素的第一个和最后一个位置

class Solution {
    public int[] searchRange(int[] nums, int target) {
        int x=left(nums,target);
        if(x==-1){
            return new int[]{-1,-1};
        }
        else{
            return new int[]{x,right(nums,target)};
        }
    }
    public int left(int[] nums,int target) {
        int i=0;
        int j=nums.length-1;
        int cand=-1;
        while(i<=j){
           int  m=(i+j)>>>1;
            if(target<nums[m]){
                j=m-1;
            }
            else if(nums[m]<target){
                i=m+1;
            }
            else{
                cand=m;
                j=m-1;
            }
        }
        return cand;
    }
     public int right(int[] nums,int target) {
        int i=0;
        int j=nums.length-1;
        int cand=-1;
        while(i<=j){
          int  m=(i+j)>>>1;
            if(target<nums[m]){
                j=m-1;
            }
            else if(nums[m]<target){
                i=m+1;
            }
            else{
                cand=m;
                i=m+1;
            }
        }
        return cand;
    }
}
目录
相关文章
|
5月前
|
前端开发 Java
java实现队列数据结构代码详解
本文详细解析了Java中队列数据结构的实现,包括队列的基本概念、应用场景及代码实现。队列是一种遵循“先进先出”原则的线性结构,支持在队尾插入和队头删除操作。文章介绍了顺序队列与链式队列,并重点分析了循环队列的实现方式以解决溢出问题。通过具体代码示例(如`enqueue`入队和`dequeue`出队),展示了队列的操作逻辑,帮助读者深入理解其工作机制。
150 1
|
3月前
|
存储 安全 Java
Java 集合面试题从数据结构到 HashMap 源码剖析详解及长尾考点梳理
本文深入解析Java集合框架,涵盖基础概念、常见集合类型及HashMap的底层数据结构与源码实现。从Collection、Map到Iterator接口,逐一剖析其特性与应用场景。重点解读HashMap在JDK1.7与1.8中的数据结构演变,包括数组+链表+红黑树优化,以及put方法和扩容机制的实现细节。结合订单管理与用户权限管理等实际案例,展示集合框架的应用价值,助你全面掌握相关知识,轻松应对面试与开发需求。
187 3
|
11月前
|
存储 人工智能 算法
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
这篇文章详细介绍了Dijkstra和Floyd算法,这两种算法分别用于解决单源和多源最短路径问题,并且提供了Java语言的实现代码。
567 3
数据结构与算法细节篇之最短路径问题:Dijkstra和Floyd算法详细描述,java语言实现。
|
8月前
|
存储 算法 C++
【C++数据结构——查找】二分查找(头歌实践教学平台习题)【合集】
二分查找的基本思想是:每次比较中间元素与目标元素的大小,如果中间元素等于目标元素,则查找成功;顺序表是线性表的一种存储方式,它用一组地址连续的存储单元依次存储线性表中的数据元素,使得逻辑上相邻的元素在物理存储位置上也相邻。第1次比较:查找范围R[0...10],比较元素R[5]:25。第1次比较:查找范围R[0...10],比较元素R[5]:25。第2次比较:查找范围R[0..4],比较元素R[2]:10。第3次比较:查找范围R[3...4],比较元素R[3]:15。,其中是顺序表中元素的个数。
288 68
【C++数据结构——查找】二分查找(头歌实践教学平台习题)【合集】
|
11月前
|
存储 Java
Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。
【10月更文挑战第19天】本文详细介绍了Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。HashMap以其高效的插入、查找和删除操作著称,而TreeMap则擅长于保持元素的自然排序或自定义排序,两者各具优势,适用于不同的开发场景。
118 1
|
11月前
|
存储 Java
告别混乱!用Java Map优雅管理你的数据结构
【10月更文挑战第17天】在软件开发中,随着项目复杂度增加,数据结构的组织和管理至关重要。Java中的Map接口提供了一种优雅的解决方案,帮助我们高效、清晰地管理数据。本文通过在线购物平台的案例,展示了Map在商品管理、用户管理和订单管理中的具体应用,有效提升了代码质量和维护性。
196 2
|
11月前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
228 2
|
5月前
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
425 1
|
9月前
|
存储 缓存 安全
Java 集合江湖:底层数据结构的大揭秘!
小米是一位热爱技术分享的程序员,本文详细解析了Java面试中常见的List、Set、Map的区别。不仅介绍了它们的基本特性和实现类,还深入探讨了各自的使用场景和面试技巧,帮助读者更好地理解和应对相关问题。
137 5
|
10月前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
164 6