X6-2、java數(shù)據(jù)結(jié)構(gòu)---二分查找算法【2020-12-22】

總目錄:地址如下看總綱

http://www.itdecent.cn/p/929ca9e209e8

1、介紹【前提有序】

二分查找又稱折半查找,既從中間數(shù)開(kāi)始找,然后根據(jù)比較結(jié)果,選擇需要折半的一邊出發(fā),
再次折半,以此類推找出元素

2、設(shè)計(jì)思路(單數(shù))

  1. 、確定當(dāng)前數(shù)組的下標(biāo) mid = (left + right)/2;
  2. 、其次讓需要查找的數(shù) findVal 和 arr[mid] 比較
    (1) findVal > arr[mid] 說(shuō)明要查找的數(shù)在 mid 右邊,因此向右遞歸查找
    (2) findVal < arr[mid] 說(shuō)明要查找的數(shù)在 mid 左邊,因此向左遞歸查找
    (3) findVal = arr[mid] 說(shuō)明找到,既返回
  3. 、需要注意何時(shí)結(jié)束遞歸?
    (1)找到就結(jié)束
    (2)遞歸完整個(gè)數(shù)組,仍然沒(méi)有找到 findVal,也需要結(jié)束遞歸 ,當(dāng) left > rigth 就是結(jié)束遞歸的條件

3、代碼(單數(shù)版):只返回一個(gè)

/**
     * 二分查找(單數(shù)版)
     * 
     * @param arr     原始數(shù)組
     * @param left    左邊索引
     * @param right   右邊索引
     * @param findVal 查找的值
     * @return
     */
    public static int binarySearch(int[] arr, int left, int right, int findVal) {

        // 當(dāng) left > right 時(shí)候,說(shuō)明已經(jīng)遞歸完畢,仍沒(méi)有找到
        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、設(shè)計(jì)思路(多數(shù))

  1. 、找到第一個(gè)匹配的索引值 mid 的時(shí)候,不要馬上返回
  2. 、向 mid 索引值 的左邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回
  3. 、向 mid 索引值 的右邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回

5、代碼(多數(shù)版):找到幾個(gè)返回幾個(gè)

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

        // 當(dāng) left > right 時(shí),說(shuō)明已經(jīng)遞歸完畢,仍沒(méi)有找到
        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 {
            // ****************以下代碼才是和簡(jiǎn)單版的區(qū)別所在,以上都一樣*******
            // 1、找到第一個(gè)匹配的索引值 mid 的時(shí)候,不要馬上返回
            // 用于接受結(jié)果
            List<Integer> resIndexList = new ArrayList();

            // 2、向 mid 索引值 的左邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回
            int temp = mid - 1;// 左遞歸起始索引
            while (true) {
                if (temp < 0 || arr[temp] != findVal) {
                    break;// 不滿足,退出(固二分為有序,左邊的第一個(gè)不相同,再找也是不同)
                } else {
                    // 加入,左移(再找)
                    resIndexList.add(temp);
                    temp -= 1;
                }
            }
            resIndexList.add(mid);

            // 3、向 mid 索引值 的右邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != findVal) {
                    break;
                } else {
                    resIndexList.add(temp);
                    temp += 1;
                }
            }
            return resIndexList;
        }
    }

6、最終完整代碼

/**
 * title: 二分查找(單,多數(shù)版)
 * @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));
    }

    /**
     * 二分查找(單數(shù)版)
     * 
     * @param arr     原始數(shù)組
     * @param left    左邊索引
     * @param right   右邊索引
     * @param findVal 查找的值
     * @return
     */
    public static int binarySearch(int[] arr, int left, int right, int findVal) {

        // 當(dāng) left > right 時(shí)候,說(shuō)明已經(jīng)遞歸完畢,仍沒(méi)有找到
        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;
        }
    }

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

        // 當(dāng) left > right 時(shí),說(shuō)明已經(jīng)遞歸完畢,仍沒(méi)有找到
        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 {
            // ****************以下代碼才是和簡(jiǎn)單版的區(qū)別所在,以上都一樣*******
            // 1、找到第一個(gè)匹配的索引值 mid 的時(shí)候,不要馬上返回
            // 用于接受結(jié)果
            List<Integer> resIndexList = new ArrayList();

            // 2、向 mid 索引值 的左邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回
            int temp = mid - 1;// 左遞歸起始索引
            while (true) {
                if (temp < 0 || arr[temp] != findVal) {
                    break;// 不滿足,退出(固二分為有序,左邊的第一個(gè)不相同,再找也是不同)
                } else {
                    // 加入,左移(再找)
                    resIndexList.add(temp);
                    temp -= 1;
                }
            }
            resIndexList.add(mid);

            // 3、向 mid 索引值 的右邊掃描,將所有與 findVal 相等的元素 的下標(biāo) 索引返回
            temp = mid + 1;
            while (true) {
                if (temp > arr.length - 1 || arr[temp] != findVal) {
                    break;
                } else {
                    resIndexList.add(temp);
                    temp += 1;
                }
            }
            return resIndexList;
        }
    }
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容