美文网首页程序员
X6-2、java数据结构---二分查找算法【2020-12-2

X6-2、java数据结构---二分查找算法【2020-12-2

作者: 鄙人_阿K | 来源:发表于2020-11-21 13:36 被阅读0次

总目录:地址如下看总纲

https://www.jianshu.com/p/929ca9e209e8

1、介绍【前提有序】

二分查找又称折半查找,既从中间数开始找,然后根据比较结果,选择需要折半的一边出发,
再次折半,以此类推找出元素

2、设计思路(单数)

  1. 、确定当前数组的下标 mid = (left + right)/2;
  2. 、其次让需要查找的数 findVal 和 arr[mid] 比较
    (1) findVal > arr[mid] 说明要查找的数在 mid 右边,因此向右递归查找
    (2) findVal < arr[mid] 说明要查找的数在 mid 左边,因此向左递归查找
    (3) findVal = arr[mid] 说明找到,既返回
  3. 、需要注意何时结束递归?
    (1)找到就结束
    (2)递归完整个数组,仍然没有找到 findVal,也需要结束递归 ,当 left > rigth 就是结束递归的条件

3、代码(单数版):只返回一个

/**
     * 二分查找(单数版)
     * 
     * @param arr     原始数组
     * @param left    左边索引
     * @param right   右边索引
     * @param findVal 查找的值
     * @return
     */
    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;
        }
    }

4、设计思路(多数)

  1. 、找到第一个匹配的索引值 mid 的时候,不要马上返回
  2. 、向 mid 索引值 的左边扫描,将所有与 findVal 相等的元素 的下标 索引返回
  3. 、向 mid 索引值 的右边扫描,将所有与 findVal 相等的元素 的下标 索引返回

5、代码(多数版):找到几个返回几个

    /**
     * 二分查找(多数版)
     * 
     * @param arr     原始数组
     * @param left    左边索引
     * @param right   右边索引
     * @param findVal 查找的值
     * @return
     */
    public static List<Integer> binarySearchS(int[] arr, int left, int right, int findVal) {

        // 当 left > right 时,说明已经递归完毕,仍没有找到
        if (left > right) {
            return new ArrayList<Integer>();
        }

        // 折半取值
        int mid = (left + right) / 2;
        int midVal = arr[mid];

        if (findVal > midVal) {
            // 右递归
            return binarySearchS(arr, mid + 1, right, findVal);
        } else if (findVal < midVal) {
            // 左递归
            return binarySearchS(arr, left, mid - 1, findVal);
        } else {
            // ****************以下代码才是和简单版的区别所在,以上都一样*******
            // 1、找到第一个匹配的索引值 mid 的时候,不要马上返回
            // 用于接受结果
            List<Integer> resIndexList = new ArrayList();

            // 2、向 mid 索引值 的左边扫描,将所有与 findVal 相等的元素 的下标 索引返回
            int temp = mid - 1;// 左递归起始索引
            while (true) {
                if (temp < 0 || arr[temp] != findVal) {
                    break;// 不满足,退出(固二分为有序,左边的第一个不相同,再找也是不同)
                } else {
                    // 加入,左移(再找)
                    resIndexList.add(temp);
                    temp -= 1;
                }
            }
            resIndexList.add(mid);

            // 3、向 mid 索引值 的右边扫描,将所有与 findVal 相等的元素 的下标 索引返回
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != findVal) {
                    break;
                } else {
                    resIndexList.add(temp);
                    temp += 1;
                }
            }
            return resIndexList;
        }
    }

6、最终完整代码

/**
 * title: 二分查找(单,多数版)
 * @author 阿K
 * 2020年12月22日 下午9:56:17 
 */
public class BinarySearch {

    public static void main(String[] args) {
        int arr[] = { 1, 8, 10, 89, 1000, 1000, 1234 };

        int findVal = 1000;
        System.out.println(binarySearch(arr, 0, arr.length - 1, findVal));
        System.out.println(binarySearchS(arr, 0, arr.length - 1, findVal));
    }

    /**
     * 二分查找(单数版)
     * 
     * @param arr     原始数组
     * @param left    左边索引
     * @param right   右边索引
     * @param findVal 查找的值
     * @return
     */
    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;
        }
    }

    /**
     * 二分查找(多数版)
     * 
     * @param arr     原始数组
     * @param left    左边索引
     * @param right   右边索引
     * @param findVal 查找的值
     * @return
     */
    public static List<Integer> binarySearchS(int[] arr, int left, int right, int findVal) {

        // 当 left > right 时,说明已经递归完毕,仍没有找到
        if (left > right) {
            return new ArrayList<Integer>();
        }

        // 折半取值
        int mid = (left + right) / 2;
        int midVal = arr[mid];

        if (findVal > midVal) {
            // 右递归
            return binarySearchS(arr, mid + 1, right, findVal);
        } else if (findVal < midVal) {
            // 左递归
            return binarySearchS(arr, left, mid - 1, findVal);
        } else {
            // ****************以下代码才是和简单版的区别所在,以上都一样*******
            // 1、找到第一个匹配的索引值 mid 的时候,不要马上返回
            // 用于接受结果
            List<Integer> resIndexList = new ArrayList();

            // 2、向 mid 索引值 的左边扫描,将所有与 findVal 相等的元素 的下标 索引返回
            int temp = mid - 1;// 左递归起始索引
            while (true) {
                if (temp < 0 || arr[temp] != findVal) {
                    break;// 不满足,退出(固二分为有序,左边的第一个不相同,再找也是不同)
                } else {
                    // 加入,左移(再找)
                    resIndexList.add(temp);
                    temp -= 1;
                }
            }
            resIndexList.add(mid);

            // 3、向 mid 索引值 的右边扫描,将所有与 findVal 相等的元素 的下标 索引返回
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != findVal) {
                    break;
                } else {
                    resIndexList.add(temp);
                    temp += 1;
                }
            }
            return resIndexList;
        }
    }
}

相关文章

  • X6-2、java数据结构---二分查找算法【2020-12-2

    总目录:地址如下看总纲 https://www.jianshu.com/p/929ca9e209e8[https:...

  • 排序算法

    算法与数据结构基础 查找算法: 二分查找法: 简介:二分查找法又被称为折半查找法,用于预排序的查找问题 过程: 如...

  • 想去阿里——这是你必备的实力

    算法和数据结构数组、链表、二叉树、队列、栈的各种操作(性能,场景) 二分查找和各种变种的二分查找 各类排序算法以及...

  • 想去阿里——这是你必备的实力

    算法和数据结构数组、链表、二叉树、队列、栈的各种操作(性能,场景)二分查找和各种变种的二分查找各类排序算法以及复杂...

  • 数据结构与算法--散列表

    数据结构与算法--散列表 之前学习了基于链表的顺序查找、基于有序数组的二分查找、二叉查找树、红黑树,这些算法在查找...

  • 二分查找

    二分查找是一种查询效率非常高的查找算法。又称折半查找。 起初在数据结构中学习递归时实现二分查找,实际上不用递归也可...

  • java实现二分查找-两种方式

    二分查找是一种查询效率非常高的查找算法。又称折半查找。 起初在数据结构中学习递归时实现二分查找,实际上不用递归也可...

  • 数据结构和算法--二分查找

    二分查找 二分查找的思想 二分查找(Binary Search)算法,也叫折半查找算法。 二分查找针对的是一个有序...

  • 数据结构与算法 树 引言 顺序查找 ​ 哨兵的方式 ​ 时间复杂度为O(N) 二分查找查找树的形式我...

  • 【爬虫】数据结构实现折半查找的算法

    数据结构实现折半查找的算法 折半查找技术,也就是二分查找,通常称为二分法查找。它的前期是线性表中的记录必须是关键码...

网友评论

    本文标题:X6-2、java数据结构---二分查找算法【2020-12-2

    本文链接:https://www.haomeiwen.com/subject/mebobktx.html