【算法】——二分查找合集

简介: 二分查找基础模版和进阶模版,查找元素位置,搜索插入位置,x的平方根,山脉数组的峰顶索引,寻找峰值,点名

  image.gif 编辑

阿华代码,不是逆风,就是我疯

你们的点赞收藏是我前进最大的动力!!

希望本文内容能够帮助到你!!

目录

零:二分查找工具

1:最基础模版

2:mid落点问题

一:最简单的二分

二:查找元素的位置(可能会有多个)

三:搜索插入位置

四:x的平方根

五:山脉数组的峰顶索引

六:寻找峰值

编辑

解法一

解法二

七:点名


零:二分查找工具

1:最基础模版

mid的写法可以防止溢出

image.gif 编辑

2:mid落点问题

image.gif 编辑

巧妙记忆:循环条件为while(left<right)时,left = mid,想象一下,只剩下两个球,那么我们的mid只能落在右端点,否则left = mid 会造成 left < right 死循环,此时我们确定的是右边的界限

重点:left 和right 根据题目的意思进行设置,然后才是mid的设置根据left和right的设置而设置(这才是这个二分查找的精髓所在)

简单记忆:落在哪个端点确定哪个界限

一:最简单的二分

704. 二分查找 - 力扣(LeetCode)

image.gif 编辑

class Solution {
    public int search(int[] nums, int target) {
        //mid=left + (right - left)/3
        //用left移动思想来确定mid的位置,这种写法可以防溢出
        int left = 0 , right = nums.length-1 , mid = (left+right)/2;
        while(left<=right){
            if(nums[mid] < target){
                left = mid + 1 ;
                mid = (left+right)/2;
            }else if(nums[mid] > target){
                right = mid - 1;
                mid = (left+right)/2;
            }else{
                return mid;
            }
        }
        return -1;
    }
}

image.gif

二:查找元素的位置(可能会有多个)

34. 在排序数组中查找元素的第一个和最后一个位置 - 力扣(LeetCode)

image.gif 编辑

image.gif 编辑

class Solution {
    public int[] searchRange(int[] nums, int target) {
        int[] result = new int[]{-1,-1};
        if(nums.length == 0 ){
            return result;
        }
        //左端点
        
        int left = 0 , right = nums.length-1 ,targetLeft = 0 , targetRight = 0;
        while(left < right){
            int mid = left + (right - left )/2;
           if(nums[mid] < target){
                left = mid + 1;
           }else{
                right = mid;
           }
        }
        targetLeft = left;
        left = 0 ; right = nums.length-1 ;
        //右端点
        while(left < right){
            int mid = left + (right-left+1)/2;
            if(nums[mid] > target){
                right = mid - 1;
            }else{
                left = mid;
            }
        }
        targetRight = right;
        if(nums[targetLeft] != target){
            return result;
        }else if(nums[targetLeft] == nums[targetRight]){
            result[0] = targetLeft;
            result[1] = targetRight;
            return result;
        }else{
        }
        
        return result;
    }
}

image.gif

三:搜索插入位置

35. 搜索插入位置 - 力扣(LeetCode)

image.gif 编辑

image.gif 编辑

image.gif 编辑

class Solution {
    public int searchInsert(int[] nums, int target) {
        if(nums.length == 0){
            return 0;
        }
        int targetLeft = 0  , n = nums.length;
        int left = 0 , right = nums.length-1;
        //这道题只用找一个左界限就够了
       
        //左界限
        left = 0 ; right = n-1;
        while(left < right){
            int mid = left + (right - left)/2;//左端点
            if(nums[mid] >= target){
                right = mid;
            }else{
                left = mid + 1;
            }
        } 
        targetLeft = left;
        int result = 0;
        
            
            if(target > nums[targetLeft]){
                result = targetLeft + 1;
            }else{
                result = targetLeft ;
            }
        
       
        
        return result;
    }
}

image.gif

四:x的平方根

69. x 的平方根 - 力扣(LeetCode)

image.gif 编辑

class Solution {
    public int mySqrt(int x) {
        long left = 0 , right = x ;
        if(x < 1 ){
            return 0;
        }
        long mid = 0;//mid的平方越界了
       while(left < right){
            mid = left + (right - left + 1)/2;
            if(mid * mid <= x){
                left = mid;
            }else{
                right = mid - 1 ;
            }
       }
       return (int)left;//强转为int类型
    }
}

image.gif

五:山脉数组的峰顶索引

852. 山脉数组的峰顶索引 - 力扣(LeetCode)

image.gif 编辑

image.gif 编辑

class Solution {
    public int peakIndexInMountainArray(int[] arr) {
        int left = 0 , right = arr.length , n = arr.length;
        while(left < right){
            int mid = left + (right - left + 1)/2;
            if(arr[mid] > arr[mid-1]){
                left = mid;
            }else if(arr[mid] < arr[mid-1]){
                right = mid - 1;
            }else{
                
            }
        }
        return left;
    }
}

image.gif

六:寻找峰值

162. 寻找峰值 - 力扣(LeetCode)

image.gif 编辑

image.gif 编辑

image.gif 编辑

解法一

class Solution {
    public int findPeakElement(int[] nums) {
        int left = 0 , right = nums.length-1;
        //如果数组中只有一个元素,while循环都进不去,规避了这个问题nb
        while(left < right){
            int mid = left + (right - left )/2;
            if(nums[mid+1] > nums[mid]){
                left = mid + 1;
            }else if(nums[mid+1] < nums[mid]){
                right = mid;
            }else{
            }
        }
        return left;
    }
}

image.gif

解法二

class Solution {
    public int findPeakElement(int[] nums) {
        //暴力解法
        int n = nums.length , result = 0;
        if(n == 1){
            result = 0;
        }else if(nums[0] > nums[1]){
            result = 0;
        }else if(nums[n-1] > nums[n-2]){
            result = n-1;
        }else{
            int left = 0 , right = nums.length ;
            while(left < right){
                int mid = left + (right - left + 1)/2;
                if(nums[mid] > nums[mid-1]){
                    left = mid;
                }else if(nums[mid] < nums[mid-1]){
                    right = mid-1;
                }else{
                }
            }
            result = left;
        }
        return result;
    }
}

image.gif

七:寻找旋转排序数组中的最小值

153. 寻找旋转排序数组中的最小值

image.gif 编辑

image.gif 编辑

class Solution {
    public int findMin(int[] nums) {
        int left = 0 , n = nums.length , right = n-1;
        int tem = nums[n-1];
        while(left < right){
            int mid = left + (right - left)/2;
            if(nums[mid] <= nums[n-1]){
                right = mid;
            }else{
                left = mid + 1;
            }
        }
        return nums[left];
    }
}

image.gif

八:点名

LCR 173. 点名 - 力扣(LeetCode)

image.gif 编辑

image.gif 编辑

class Solution {
    public int takeAttendance(int[] records) {
        int left = 0 , n = records.length , right = records.length-1;
        if(records[0] != 0){
            return 0;
        }
        if(records[n-1] == n-1){
            return n;
        }
        while(left < right){
            int mid = left + (right - left)/2;
            if(records[mid] - mid <= 0){
                left = mid + 1;
            }else{
                right = mid ;
            }
        }
        return right;
    }
}

image.gif


相关文章
|
5月前
|
算法
【算法】二分查找——在排序数组中查找元素的第一个和最后一个位置
【算法】二分查找——在排序数组中查找元素的第一个和最后一个位置
|
3月前
|
算法 C# 索引
C#二分查找算法
C#二分查找算法
|
3月前
|
存储 算法 C语言
【C语言】二分查找算法
【C语言】二分查找算法
|
3月前
|
消息中间件 存储 算法
一文搞懂二分查找算法!
一文搞懂二分查找算法!
157 0
|
3月前
|
算法 Java 索引
数据结构与算法学习十五:常用查找算法介绍,线性排序、二分查找(折半查找)算法、差值查找算法、斐波那契(黄金分割法)查找算法
四种常用的查找算法:顺序查找、二分查找(折半查找)、插值查找和斐波那契查找,并提供了Java语言的实现代码和测试结果。
40 0
|
5月前
|
存储 算法 Java
深入算法基础二分查找数组
文章深入学习了二分查找算法的基础,通过实战例子详细解释了算法的逻辑流程,强调了确定合法搜索边界的重要性,并提供了Java语言的代码实现。
深入算法基础二分查找数组
|
6月前
|
算法
【算法】二分查找(整数二分和浮点数二分)
算法学习——二分查找(整数二分和浮点数二分)
52 0
【算法】二分查找(整数二分和浮点数二分)
|
5月前
|
算法
【算法】二分查找——二分查找
【算法】二分查找——二分查找
|
7月前
|
存储 算法 C语言
二分查找算法的概念、原理、效率以及使用C语言循环和数组的简单实现
二分查找算法的概念、原理、效率以及使用C语言循环和数组的简单实现
|
7月前
|
机器学习/深度学习 算法 索引
数据结构算法--1 顺序查找二分查找
**顺序查找时间复杂度为O(n)**,适合无序列表,可以通过`enumerate`或直接遍历索引来实现。**二分查找时间复杂度为O(logn)**,适用于有序列表,利用Python中`left`、`right`指针和`mid`点不断缩小搜索范围。效率上二分查找更优。