首页 > 其他分享 >二分查找步骤及问题总结

二分查找步骤及问题总结

时间:2022-09-23 18:56:47浏览次数:48  
标签:二分 arr right target int 步骤 mid 查找 left

二分查找

参数: 有序数组arr(这里按升序来讲),待搜索的值target

步骤

  1. 定义左边界left和有边界right
  2. 获取中间索引(整数) mid = (left+right)/2,注意:js只有小数,mid需要再取整
  3. 中间索引的值arr[mid]与待搜索的值target进行比较
    • arr[mid] == target ,即为找到,返回中间索引mid
    • arr[mid] > target,说明要搜索的值在mid的左边(降序情况相反),需要去mid的左边找,更改右边界right为mid-1,重新查找
    • arr[mid] < target,说明要搜索的值在mid的右边(降序情况相反),需要去mid的右边找,更改左边界left为mid+1,重新查找
  4. 查找(循环)的过程中如果left > right说明找不到target,结束查找

根据以上步骤可以写出递归、和非递归两种二分查找的方法

代码实现

    public static void main(String[] args) {
        int arr[] = {1, 8, 10, 11,23,1000,1234};
        //int resIndex = binarySearch(arr, 0, arr.length - 1, 11);
        int resIndex = binarySearch(arr,11);
        System.out.println(resIndex);
    }
	


	/** 非递归二分查找
     * @param arr    待查找的数组(升序
     * @param target 需要查找的值
     * @return 返回对应下标,-1表示没有找到
     */
    public static int binarySearchNoRecur(int[] arr, int target) {
        int left = 0;
        int right = arr.length - 1;
        while (left <= right) { //说明需要继续查找
            int mid = (left + right) / 2;
            if (arr[mid] == target) {
                return mid;
            } else if (arr[mid] > target) {
                right = mid - 1;//向mid的左边
            } else {
                left = mid + 1;//向mid 的右边
            }
        }
        return -1;
    }

	/** 递归版二分查找
     * @param arr    待查找的数组(升序
     * @param target 需要查找的值
     * @return 返回对应下标,-1表示没有找到
     */
     public static int binarySearch(int[] arr, int left, int right, int target) {

        if (left > right) {
            return -1;
        }
        int mid = (left + right) / 2;
        int midVal = arr[mid];
        if (target > midVal) { //向右递归
            return binarySearch(arr, mid + 1, right, target);
        } else if (target < midVal) { //向左递归
            return binarySearch(arr, left, mid - 1, target);
        } else {//找到
            return mid;
        }
    }

	

整数溢出问题

出现原因:java 中的 int 总共就 32 位,正数上限的情况首位也只能是 0,其他位都可以是 1(就是 2^31-1 的情况)。但是如果正数过大了,例如 2^31,计算机不得不把首位变成 1,把它按照正常的方式输出了(把1作为符号位),于是就成了负的值。

获取中间索引时(left + right) / 2,如果left和right都特别大,那么就有可能超出整数锁能存储的最大值,从而出现整数溢出问题

如下情况,第二次的mid出现了整数溢出问题

	int left = 0;
        int right = Integer.MAX_VALUE - 1;
        int mid = (left+right)/2;
        System.out.println(mid);
        //如果中间值小于目标值,需要更改左边界为mid+1
		left = mid + 1;
        mid = (left+right)/2;
        System.out.println(mid);

输出

1073741823
-536870913

如何避免?

方法一

把求中间值公式改为left - left/2 + right/2 =》left + (right - left)/2

由于两个大值相减(right - left)得到的值较小,就避免了溢出问题

方法二

无符号的右移运算代替除法

(left+right)/2 =》(left+right)>>>1

本来溢出的二进制数符号位为负数,由于进行了移位,就不把最高位当成符号位,就解决了溢出问题

标签:二分,arr,right,target,int,步骤,mid,查找,left
From: https://www.cnblogs.com/weloe/p/16723894.html

相关文章

  • Anaconda功能、优点、安装步骤(安装视频)
    目录介绍功能(包和环境的管理器)优点(省时省心)下载地址安装教程要点conda的常见命令查询完整帮助文件管理conda和anaconda管理环境包管理其他​介绍Anaconda是专注于数据......
  • 二分模板
    intsearch(vector<int>&nums,inttarget){intleft=0,right=nums.size();intmid;while(left<right){mid=(left+right)>>1;......
  • 快速稳定盈利的四个步骤
    控制自己不会有大幅度的回撤是稳定盈利的第一步。如何去控制回撤呢?第一是你要记录你对所有模式的实操情况。然后记录下来。比如你用半路战法的时候最大的一次日回撤是多少......
  • 459测试概述和460junit使用步骤
    测试概述单元测试:1.黑盒测试:不需要写代码,输入值,看程序是否能够输出期望值2.白盒测试:需要写代码。关注程序具体执行的流程  Junit使用步骤白盒测试步骤:1.定......
  • 实例84 二分法求解方程
    #include<stdio.h>#include<math.h>#include<malloc.h>#include<stdlib.h>doubleFunc(double);intBisectRoot(double,double,double,double,double*,int,in......
  • sql练习--查找山东大学或者性别为男生的信息
    描述题目:现在运营想要分别查看学校为山东大学或者性别为男性的用户的device_id、gender、age和gpa数据,请取出相应结果,结果不去重。 示例:user_profileiddevice_i......
  • Junit使用步骤和Junit-@Before@Afte
    Junit使用步骤白盒测试步骤:1.定义一个测试类(测试用例)建议:测试类名:被测试的类名Test包名:xxx.xxx.xx.test2.定义测试方法......
  • Python3交叉编译步骤(二)-三方库的交叉编译
    一.项目场景在cortex-A9主板上运行python3,能够使用常用的三方库二.配置主机环境:ubuntu-18.04-x86_64(虚拟机)交叉编译链:arm-linux-gnueabihf-gcc开发板:cortex-A9(armv7l)三.......
  • React + Eartho 与 3 个简单的步骤集成
    React+Eartho与3个简单的步骤集成如果您已经关注并访问了您的第一个地球和React经验,那么我相信你会感觉很棒。它一次为开发人员提供了许多好处。如果你有地球......
  • 代码随想录 数组理论基础,二分查找,移除元素 及 LeetCode 704, 27
    数组理论基础数组是存放在连续内存空间上的相同类型数据的集合。数组下标都是从0开始的。数组内存空间的地址是连续的因为数组的在内存空间的地址是连续的,所以我们在......