二分查找算法递归方式并完善

算法思想

又叫折半查找,要求待查找的序列有序。每次取中间位置的值与待查关键字比较,如果中间位置的值比待查关键字大,则在前半部分循环这个查找的过程,如果中间位置的值比待查关键字小,则在后半部分循环这个查找的过程。直到查找到了为止,否则序列中没有待查的关键字。
折半查找法的优点是比较次数少,查找速度快,平均性能好;其缺点是要求待查表为有序表,且插入删除困难。因此,折半查找方法适用于不经常变动而查找频繁的有序列表。

二分算法步骤描述

① 首先确定整个查找区间的中间位置 mid = ( left + right )/ 2
② 用待查关键字值与中间位置的关键字值进行比较;

若相等,则查找成功

若大于,则在后(右)半个区域继续进行折半查找

若小于,则在前(左)半个区域继续进行折半查找

③ 对确定的缩小区域再按折半公式,重复上述步骤。

最后,得到结果:要么查找成功, 要么查找失败。折半查找的存储结构采用一维数组存放。

二分查找算法讨论:

优点

ASL≤log2n,即每经过一次比较,查找范围就缩小一半。经log2n 次计较就可以完成查找过程。

缺点

因要求有序,所以要求查找数列必须有序,而对所有数据元素按大小排序是非常费时的操作。另外,顺序存储结构的插入、删除操作不便利。

二分查找代码

public class binarySearch {
    public static void main(String[] args) {
        int arr[] ={1,2,3,4,1000,1000,1000,1005,1008};

        int resIndex = BinarySearch(arr,0,arr.length-1,1008);
        System.out.println("下标为"+resIndex);
    }
    public static int BinarySearch(int[] arr,int left,int right,int findVal){
        int mid = (left + right)/2;
        System.out.println("下标"+mid);
        int midVal=arr[mid];
        System.out.println("值"+midVal);
        if (findVal>midVal){ //向右递归
            return BinarySearch(arr,mid+1,right,findVal);
        }else if (findVal<midVal){ //向左递归
            return BinarySearch(arr,left,mid-1,findVal);
        }else{ //找到
            return mid;
        }
    }

}

缺陷

该代码如果查找数组中不存在的数会进入死循环

改进方法

当left>right时,说明递归整个数组,但是没有找到。如下:

public class binarySearch {
    public static void main(String[] args) {
        int arr[] ={1,2,3,4,1000,1000,1000,1005,1008};

        int resIndex = BinarySearch(arr,0,arr.length-1,1018);
        System.out.println("下标为"+resIndex);
    }
    public static int BinarySearch(int[] arr,int left,int right,int findVal){
        //当left>right时,说明递归整个数组,但是没有找到
        if(left>right){
            return -1;
        }
        int mid = (left + right)/2;
        System.out.println("下标"+mid);
        int midVal=arr[mid];
        System.out.println("值"+midVal);
        if (findVal>midVal){ //向右递归
            return BinarySearch(arr,mid+1,right,findVal);
        }else if (findVal<midVal){ //向左递归
            return BinarySearch(arr,left,mid-1,findVal);
        }else{ //找到
            return mid;
        }
    }

}

提出问题❓

如果集合中存在多个相同的数呢?

改进思路

思路分析
	1.在找到mid索引值,不需要马上返回
	2.向mid索引值的左边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
	3.向mid索引值的右边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
	4.将ArrayList返回

代码实现

/**
 * 前提:数组需要是有序数组
 * 二分查找(对分查找)
 * *** 解决了数组中有多个相同的数,也能找到这些数对应的多个下标
 * 提示:可以通过返回集合的Size判断有没有找到数值
 *
 * @Author Yren
 */
public class binarySearch {
    public static void main(String[] args) {
        int arr[] = {1, 2, 3, 4, 1000, 1000, 1000, 1000, 1005, 1008};

//        int resIndex = BinarySearch(arr, 0, arr.length - 1, 1018);
//        System.out.println("下标为" + resIndex);
        List<Integer> resIndexList = BinarySearch2(arr, 0, arr.length - 1, 1000);
        System.out.println("resIndexList=" + resIndexList);
    }

    public static int BinarySearch(int[] arr, int left, int right, int findVal) {
        //当left>right时,说明递归整个数组,但是没有找到
        if (left > right) {
            return -1;
        }
        int mid = (left + right) / 2;
        int midVal = arr[mid];
        if (findVal > midVal) { //向右递归
            return BinarySearch(arr, mid + 1, right, findVal);
        } else if (findVal < midVal) { //向左递归
            return BinarySearch(arr, left, mid - 1, findVal);
        } else { //找到
            return mid;
        }
    }

    public static ArrayList<Integer> BinarySearch2(int[] arr, int left, int right, int findVal) {
        //当left>right时,说明递归整个数组,但是没有找到
        if (left > right) {
            return new ArrayList<Integer>();
        }
        int mid = (left + right) / 2;
        System.out.println("下标" + mid);
        int midVal = arr[mid];
        System.out.println("值" + midVal);
        if (findVal > midVal) { //向右递归
            return BinarySearch2(arr, mid + 1, right, findVal);
        } else if (findVal < midVal) { //向左递归
            return BinarySearch2(arr, left, mid - 1, findVal);
        } else { //找到
            /**思路分析
             * 1.在找到mid索引值,不需要马上返回
             * 2.向mid索引值的左边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
             * 3.向mid索引值的右边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
             * 4.将ArrayList返回
             */
            ArrayList<Integer> resIndexlist = new ArrayList<Integer>();
            //向mide索引值的左边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
            int temp = mid - 1;
            while (true) {
                if (temp < 0 || arr[temp] != findVal) {
                    //temp<0说明已经扫描到最左边了 || arr[temp] != findVal条件满足说明遍历过程中发现了一个不等于findVal的值,那就可以退出了因为想同的值是连续存放的
                    break;
                }
                //否则,就temp 放入到resIndexlist
                resIndexlist.add(temp);
                temp -= 1;//temp左移
            }
            resIndexlist.add(mid);

            //向mide索引值的右边扫描,将所有满足1000,的元素的下标,加入到集合ArrayList
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != findVal) {
                    //temp<0说明右边已经扫描完了 || arr[temp] != findVal条件满足说明遍历右边过程中发现了一个不等于findVal的值,那就可以退出了因为想同的值是连续存放的
                    break;
                }
                //否则,就temp 放入到resIndexlist
                resIndexlist.add(temp);
                temp += 1;//temp右移
            }
            return resIndexlist;
        }
    }

}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值