色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

時間復雜度為O (nlogn)的排序算法簡述

OSC開源社區 ? 來源:京東云開發者-京東物流 ? 2023-12-05 09:57 ? 次閱讀

歸并排序

歸并排序遵循分治的思想:將原問題分解為幾個規模較小但類似于原問題的子問題,遞歸地求解這些子問題,然后合并這些子問題的解來建立原問題的解,歸并排序的步驟如下:

劃分:分解待排序的 n 個元素的序列成各具 n/2 個元素的兩個子序列,將長數組的排序問題轉換為短數組的排序問題,當待排序的序列長度為 1 時,遞歸劃分結束

合并:合并兩個已排序的子序列得出已排序的最終結果

歸并排序的代碼實現如下:

    private void sort(int[] nums, int left, int right) {
        if (left >= right) {
            return;
        }

        // 劃分
        int mid = left + right >> 1;
        sort(nums, left, mid);
        sort(nums, mid + 1, right);
        // 合并
        merge(nums, left, mid, right);
    }

    private void merge(int[] nums, int left, int mid, int right) {
        // 輔助數組
        int[] temp = Arrays.copyOfRange(nums, left, right + 1);

        int leftBegin = 0, leftEnd = mid - left;
        int rightBegin = leftEnd + 1, rightEnd = right - left;
        for (int i = left; i <= right; i++) {
            if (leftBegin > leftEnd) {
                nums[i] = temp[rightBegin++];
            } else if (rightBegin > rightEnd || temp[leftBegin] < temp[rightBegin]) {
                nums[i] = temp[leftBegin++];
            } else {
                nums[i] = temp[rightBegin++];
            }
        }
    }
歸并排序最吸引人的性質是它能保證將長度為 n 的數組排序所需的時間和 nlogn 成正比;它的主要缺點是所需的額外空間和 n 成正比。 算法特性:

空間復雜度:借助輔助數組實現合并,使用 O (n) 的額外空間;遞歸深度為 logn,使用 O (logn) 大小的棧幀空間。忽略低階部分,所以空間復雜度為 O (n)

非原地排序

穩定排序

非自適應排序

以上代碼是歸并排序常見的實現,下面我們來一起看看歸并排序的優化策略:

將多次創建小數組的開銷轉換為只創建一次大數組

在上文實現中,我們在每次合并兩個有序數組時,即使是很小的數組,我們都會創建一個新的 temp [] 數組,這部分耗時是歸并排序運行時間的主要部分。更好的解決方案是將 temp [] 數組定義成 sort () 方法的局部變量,并將它作為參數傳遞給 merge () 方法,實現如下:

    private void sort(int[] nums, int left, int right, int[] temp) {
        if (left >= right) {
            return;
        }

        // 劃分
        int mid = left + right >> 1;
        sort(nums, left, mid, temp);
        sort(nums, mid + 1, right, temp);
        // 合并
        merge(nums, left, mid, right, temp);
    }

    private void merge(int[] nums, int left, int mid, int right, int[] temp) {
        System.arraycopy(nums, left, temp, left, right - left + 1);
        int l = left, r = mid + 1;
        for (int i = left; i <= right; i++) {
            if (l > mid) {
                nums[i] = temp[r++];
            } else if (r > right || temp[l] < temp[r]) {
                nums[i] = temp[l++];
            } else {
                nums[i] = temp[r++];
            }
        }
    }

當數組有序時,跳過 merge () 方法 我們可以在執行合并前添加判斷條件:如果nums[mid] <= nums[mid + 1]?時我們認為數組已經是有序的了,那么我們就跳過 merge () 方法。它不影響排序的遞歸調用,但是對任意有序的子數組算法的運行時間就變成線性的了,代碼實現如下:
    private void sort(int[] nums, int left, int right, int[] temp) {
        if (left >= right) {
            return;
        }

        // 劃分
        int mid = left + right >> 1;
        sort(nums, left, mid, temp);
        sort(nums, mid + 1, right, temp);
        // 合并
        if (nums[mid] > nums[mid + 1]) {
            merge(nums, left, mid, right, temp);
        }
    }

    private void merge(int[] nums, int left, int mid, int right, int[] temp) {
        System.arraycopy(nums, left, temp, left, right - left + 1);
        int l = left, r = mid + 1;
        for (int i = left; i <= right; i++) {
            if (l > mid) {
                nums[i] = temp[r++];
            } else if (r > right || temp[l] < temp[r]) {
                nums[i] = temp[l++];
            } else {
                nums[i] = temp[r++];
            }
        }
    }

對小規模子數組使用插入排序 對小規模數組進行排序會使遞歸調用過于頻繁,而使用插入排序處理小規模子數組一般可以將歸并排序的運行時間縮短 10% ~ 15%,代碼實現如下:
    /**
     * M 取值在 5 ~ 15 之間大多數情況下都能令人滿意
     */
    private final int M = 9;

    private void sort(int[] nums, int left, int right) {
        if (left + M >= right) {
            // 插入排序
            insertSort(nums);
            return;
        }

        // 劃分
        int mid = left + right >> 1;
        sort(nums, left, mid);
        sort(nums, mid + 1, right);
        // 合并
        merge(nums, left, mid, right);
    }

    /**
     * 插入排序
     */
    private void insertSort(int[] nums) {
        for (int i = 1; i < nums.length; i++) {
            int base = nums[i];

            int j = i - 1;
            while (j >= 0 && nums[j] > base) {
                nums[j + 1] = nums[j--];
            }
            nums[j + 1] = base;
        }
    }

    private void merge(int[] nums, int left, int mid, int right) {
        // 輔助數組
        int[] temp = Arrays.copyOfRange(nums, left, right + 1);

        int leftBegin = 0, leftEnd = mid - left;
        int rightBegin = leftEnd + 1, rightEnd = right - left;
        for (int i = left; i <= right; i++) {
            if (leftBegin > leftEnd) {
                nums[i] = temp[rightBegin++];
            } else if (rightBegin > rightEnd || temp[leftBegin] < temp[rightBegin]) {
                nums[i] = temp[leftBegin++];
            } else {
                nums[i] = temp[rightBegin++];
            }
        }
    }

快速排序 快速排序也遵循分治的思想,它與歸并排序不同的是,快速排序是原地排序,而且快速排序會先排序當前數組,再對子數組進行排序,它的算法步驟如下:

哨兵劃分:選取數組中最左端元素為基準數,將小于基準數的元素放在基準數左邊,將大于基準數的元素放在基準數右邊

排序子數組:將哨兵劃分的索引作為劃分左右子數組的分界,分別對左右子數組進行哨兵劃分和排序

快速排序的代碼實現如下:

private void sort(int[] nums, int left, int right) {

        if (left >= right) {
            return;
        }

        // 哨兵劃分
        int partition = partition(nums, left, right);

        // 分別排序兩個子數組
        sort(nums, left, partition - 1);
        sort(nums, partition + 1, right);
    }

    /**
     * 哨兵劃分
     */
    private int partition(int[] nums, int left, int right) {
        // 以 nums[left] 作為基準數,并記錄基準數索引
        int originIndex = left;
        int base = nums[left];

        while (left < right) {
            // 從右向左找小于基準數的元素
            while (left < right && nums[right] >= base) {
                right--;
            }
            // 從左向右找大于基準數的元素
            while (left < right && nums[left] <= base) {
                left++;
            }
            swap(nums, left, right);
        }
        // 將基準數交換到兩子數組的分界線
        swap(nums, originIndex, left);

        return left;
    }

    private void swap(int[] nums, int left, int right) {
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right] = temp;
    }算法特性:

時間復雜度:平均時間復雜度為 O (nlogn),最差時間復雜度為 O (n2)

空間復雜度:最差情況下,遞歸深度為 n,所以空間復雜度為 O (n)

原地排序

非穩定排序

自適應排序

歸并排序的時間復雜度一直是 O (nlogn),而快速排序在最壞的情況下時間復雜度為 O (n2),為什么歸并排序沒有快速排序應用廣泛呢? 答:因為歸并排序是非原地排序,在合并階段需要借助非常量級的額外空間

快速排序有很多優點,但是在哨兵劃分不平衡的情況下,算法的效率會比較低效。下面是對快速排序排序優化的一些方法:

切換到插入排序

對于小數組,快速排序比插入排序慢,快速排序的 sort () 方法在長度為 1 的子數組中也會調用一次,所以,在排序小數組時切換到插入排序排序的效率會更高,如下:

    /**
     * M 取值在 5 ~ 15 之間大多數情況下都能令人滿意
     */
    private final int M = 9;

    public void sort(int[] nums, int left, int right) {
        // 小數組采用插入排序
        if (left + M >= right) {
            insertSort(nums);
            return;
        }

        int partition = partition(nums, left, right);
        sort(nums, left, partition - 1);
        sort(nums, partition + 1, right);
    }

    /**
     * 插入排序
     */
    private void insertSort(int[] nums) {
        for (int i = 1; i < nums.length; i++) {
            int base = nums[i];

            int j = i - 1;
            while (j >= 0 && nums[j] > base) {
                nums[j + 1] = nums[j--];
            }
            nums[j + 1] = base;
        }
    }

    private int partition(int[] nums, int left, int right) {
        int originIndex = left;
        int base = nums[left];

        while (left < right) {
            while (left < right && nums[right] >= base) {
                right--;
            }
            while (left < right && nums[left] <= base) {
                left++;
            }
            swap(nums, left, right);
        }
        swap(nums, left, originIndex);

        return left;
    }

    private void swap(int[] nums, int left, int right) {
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right] = temp;
    }

基準數優化 如果數組為倒序的情況下,選擇最左端元素為基準數,那么每次哨兵劃分會導致右數組長度為 0,進而使快速排序的時間復雜度為 O (n2),為了盡可能避免這種情況,我們可以對基準數的選擇進行優化,采用三取樣切分的方法:選取數組最左端、中間和最右端這三個值的中位數為基準數,這樣選擇的基準數大概率不是區間的極值,時間復雜度為 O (n2) 的概率大大降低,代碼實現如下:
    public void sort(int[] nums, int left, int right) {
        if (left >= right) {
            return;
        }

        // 基準數優化
        betterBase(nums, left, right);

        int partition = partition(nums, left, right);

        sort(nums, left, partition - 1);
        sort(nums, partition + 1, right);
    }

    /**
     * 基準數優化,將 left, mid, right 這幾個值中的中位數換到 left 的位置
     * 注意其中使用了異或運算進行條件判斷
     */
    private void betterBase(int[] nums, int left, int right) {
        int mid = left + right >> 1;

        if ((nums[mid] < nums[right]) ^ (nums[mid] < nums[left])) {
            swap(nums, left, mid);
        } else if ((nums[right] < nums[left]) ^ (nums[right] < nums[mid])) {
            swap(nums, left, right);
        }
    }

    private int partition(int[] nums, int left, int right) {
        int originIndex = left;
        int base = nums[left];

        while (left < right) {
            while (left < right && nums[right] >= base) {
                right--;
            }
            while (left < right && nums[left] <= base) {
                left++;
            }
            swap(nums, left, right);
        }
        swap(nums, originIndex, left);

        return left;
    }

    private void swap(int[] nums, int left, int right) {
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right] = temp;
    }

三向切分 在數組有大量重復元素的情況下,快速排序的遞歸性會使元素全部重復的子數組經常出現,而對這些數組進行快速排序是沒有必要的,我們可以對它進行優化。 一個簡單的想法是將數組切分為三部分,分別對應小于、等于和大于基準數的數組,每次將其中 “小于” 和 “大于” 的數組進行排序,那么最終也能得到排序的結果,這種策略下我們不會對等于基準數的子數組進行排序,提高了排序算法的效率,它的算法流程如下: 從左到右遍歷數組,維護指針 l 使得 [left, l - 1] 中的元素都小于基準數,維護指針 r 使得 [r + 1, right] 中的元素都大于基準數,維護指針 mid 使得 [l, mid - 1] 中的元素都等于基準數,其中 [mid, r] 區間中的元素還未確定大小關系,圖示如下:

d24b03b0-9035-11ee-939d-92fbcf53809c.jpg

它的代碼實現如下:
    public void sort(int[] nums, int left, int right) {
        if (left >= right) {
            return;
        }

        // 三向切分
        int l = left, mid = left + 1, r = right;
        int base = nums[l];
        while (mid <= r) {
            if (nums[mid] < base) {
                swap(nums, l++, mid++);
            } else if (nums[mid] > base) {
                swap(nums, mid, r--);
            } else {
                mid++;
            }
        }

        sort(nums, left, l - 1);
        sort(nums, r + 1, right);
    }

    private void swap(int[] nums, int left, int right) {
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right] = temp;
    }

這也是經典的荷蘭國旗問題,因為這就好像用三種可能的主鍵值將數組排序一樣,這三種主鍵值對應著荷蘭國旗上的三種顏色

審核編輯:黃飛

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 算法
    +關注

    關注

    23

    文章

    4607

    瀏覽量

    92840
  • 數組
    +關注

    關注

    1

    文章

    417

    瀏覽量

    25939

原文標題:時間復雜度為O (nlogn)的排序算法

文章出處:【微信號:OSC開源社區,微信公眾號:OSC開源社區】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    C語言實現十大經典排序算法

    比較類排序:通過比較來決定元素間的相對次序,由于其時間復雜度不能突破O(nlogn),因此也稱為非線性
    發表于 06-25 10:23 ?397次閱讀
    C語言實現十大經典<b class='flag-5'>排序</b><b class='flag-5'>算法</b>

    時間復雜度 O(n^2) 的排序算法

    作者:京東保險 王奕龍 對于小規模數據,我們可以選用時間復雜度 O(n2) 的排序算法。因為
    的頭像 發表于 10-19 16:31 ?1138次閱讀
    <b class='flag-5'>時間</b><b class='flag-5'>復雜度</b><b class='flag-5'>為</b> <b class='flag-5'>O</b>(n^2) 的<b class='flag-5'>排序</b><b class='flag-5'>算法</b>

    時間復雜度是指什么

    原理->微機原理->軟件工程,編譯原理,數據庫數據結構1.時間復雜度時間復雜度是指執行算法所需要的計算工作量,因為整個
    發表于 07-22 10:01

    各種排序算法時間空間復雜度、穩定性

    各種排序算法時間空間復雜度、穩定性一、排序算法分類:二、
    發表于 12-21 07:48

    常用的非比較排序算法:計數排序,基數排序,桶排序的詳細資料概述

    這篇文章中我們來探討一下常用的非比較排序算法:計數排序,基數排序,桶排序。在一定條件下,它們的時間
    的頭像 發表于 06-18 15:11 ?7125次閱讀
    常用的非比較<b class='flag-5'>排序</b><b class='flag-5'>算法</b>:計數<b class='flag-5'>排序</b>,基數<b class='flag-5'>排序</b>,桶<b class='flag-5'>排序</b>的詳細資料概述

    常用排序算法分析

    一種是比較排序時間復雜度O(nlogn) ~ O(n^2),主要有:冒泡
    的頭像 發表于 07-13 16:13 ?2159次閱讀

    如何求遞歸算法時間復雜度

    那么我通過一道簡單的面試題,模擬面試的場景,來帶大家逐步分析遞歸算法時間復雜度,最后找出最優解,來看看同樣是遞歸,怎么就寫成了O(n)的代碼。
    的頭像 發表于 07-13 11:30 ?2260次閱讀

    如何求遞歸算法時間復雜度

    相信很多同學對遞歸算法時間復雜度都很模糊,那么這篇Carl來給大家通透的講一講。
    的頭像 發表于 07-13 11:33 ?1605次閱讀

    常見機器學習算法的計算復雜度

    時間復雜度不是測量一個算法或一段代碼在某個機器或者條件下運行所花費的時間時間復雜度一般指
    發表于 10-02 12:45 ?812次閱讀

    動圖演示C語言10大經典排序算法(含代碼)

    本文將通過 動態演示+代碼 的形式系統地總結十大經典排序算法排序算法 算法分類 十種常見排序
    的頭像 發表于 02-07 01:24 ?731次閱讀

    算法時空復雜度分析實用指南2

    類似的,想想之前說的數據結構擴容的場景,也許`N`次操作中的某一次操作恰好觸發了擴容,導致時間復雜度提高,但總的時間復雜度依然保持在`O(N
    的頭像 發表于 04-12 14:38 ?528次閱讀
    <b class='flag-5'>算法</b>時空<b class='flag-5'>復雜度</b>分析實用指南2

    算法時空復雜度分析實用指南(上)

    本文會篇幅較長,會涵蓋如下幾點: 1、Big O 表示法的幾個基本特點。 2、非遞歸算法中的時間復雜度分析。 3、數據結構 API 的效率衡量方法(攤還分析)。
    的頭像 發表于 04-19 10:34 ?811次閱讀
    <b class='flag-5'>算法</b>時空<b class='flag-5'>復雜度</b>分析實用指南(上)

    算法時空復雜度分析實用指南(下)

    Big O 表示法的幾個基本特點。 2、非遞歸算法中的時間復雜度分析。 3、數據結構 API 的效率衡量方法(攤還分析)。 4、遞歸
    的頭像 發表于 04-19 10:35 ?688次閱讀
    <b class='flag-5'>算法</b>時空<b class='flag-5'>復雜度</b>分析實用指南(下)

    常見排序算法分類

    本文將通過動態演示+代碼的形式系統地總結十大經典排序算法排序算法 算法分類 —— 十種常見排序
    的頭像 發表于 06-22 14:49 ?906次閱讀
    常見<b class='flag-5'>排序</b><b class='flag-5'>算法</b>分類

    如何計算時間復雜度

    來完成,那么該算法的用處就不會太大。同樣如果該算法需要若干個GB的內存,那么在大部分機器上都無法使用。 一個算法的評價主要從時間復雜度和空間
    的頭像 發表于 10-13 11:19 ?2972次閱讀
    如何計算<b class='flag-5'>時間</b><b class='flag-5'>復雜度</b>
    主站蜘蛛池模板: 国产亚洲精品在浅麻豆| 无羞耻肉动漫在线观看| 521人成a天堂v| 和尚扒开双腿蹂躏| 亚在线观看免费视频入口| 东京热影院| 日本成熟bbxxxxxxxx| 99久久99久久免费精品蜜桃| 久久久中日AB精品综合| 亚洲人女同志video| 韩日午夜在线资源一区二区| 洗濯屋H纯肉动漫在线观看| 风流少妇BBWBBW69视频| 青青久久久| RUNAWAY韩国动漫免费官网版| 男人扒开添女人屁股| 91久久夜色精品| 蜜芽手机在线观看| 5g在线视讯年龄确认海外禁止进入| 久久久中日AB精品综合| 中文字幕无码他人妻味| 久青草国产观看在线视频| 中文字幕亚洲第一页| 啦啦啦 中文 中国 免费 高清在线| 再插深点嗯好大好爽| 久久亚洲A片COM人成A| 中国比基尼美女| 麻豆影视在线直播观看免费| 最近日本免费观看MV免费| 李亚男三级| 99re8热视频这在线视频| 免费毛片a在线观看67194| 99久久香蕉| 日本久久免费大片| 动漫护士被乳羞羞漫| 双性大乳浪受噗呲噗呲h总| 国产精品免费一区二区三区视频| 污污内射在线观看一区二区少妇| 国产麻豆福利AV在线观看| 亚洲狠狠网站色噜噜| 黄页网址大全免费观看|