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

总目录:地址如下看总纲

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;
        }
    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,287评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,346评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,277评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,132评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,147评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,106评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,019评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,862评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,301评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,521评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,682评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,405评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,996评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,651评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,803评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,674评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,563评论 2 352

推荐阅读更多精彩内容