iT邦幫忙

2021 iThome 鐵人賽

DAY 20
0
自我挑戰組

每日攝取一點資料結構和演算法系列 第 20

Day20:[排序演算法]Selection Sort - 選擇排序法

  • 分享至 

  • xImage
  •  

https://ithelp.ithome.com.tw/upload/images/20210917/20128604GxN0TZXzmy.jpg
在了解快速排序法的概念之前要先理解partition演算法,不過單用文字敘述還是蠻抽象的,所以搭配示意圖來做說明,假如現在有個陣列[2, 6, 3, 9, 1, 5]

  1. 取陣列的最後一個值5作為pivot基準值
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604s5VOKu0WcL.png

  2. 接下來從index 0開始, 2小於5所以標記為綠色
    https://ithelp.ithome.com.tw/upload/images/20210917/201286042YAKIjht4v.png

  3. 再來是6,6大於5所以標記為橘色
    https://ithelp.ithome.com.tw/upload/images/20210917/201286046h1DeUu6ZR.png

  4. 輪到3,發現比5小所以把3標記為綠色,因為要把比5小的值都集中到左邊,所以跟6交換位置
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604bMXLPL0emI.png

  5. 接下來是9, 比5大標記成橘色
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604K6jSWHZCjX.png

  6. 輪到1,比5小所以把1標記為綠色,因為比5小的值都要集中到左邊,所以要跟6交換位置
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604QimLnGu8dB.png

  7. 最後再讓支點5與最大值群組(橘色的便條紙)中的第一個值做交換,5這個數字就完成了排序
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604TkoLmwALCX.png

  8. 不過這時你會發現綠色的陣列並沒有由小到大排序好,這邊假設我們也不知道橘色陣列是否有排序好, 所以接下來也對左右兩個陣列,利用遞迴的方式持續做一樣的動作。
    https://ithelp.ithome.com.tw/upload/images/20210917/20128604YtPGObumPh.png

函式內部實作為:

  • 先將陣列裡的最後一個元素定義為基準值
  • 接下來傳入起始點和終點來指定迴圈跑的範圍
  • 每個遍歷的值跟基準值做比較,假如比基準值小的話就跟基準值較大的群組中最左邊的那個值交換位置,這個動作是要將比基準值小的數字都集中在左邊

進行完一輪之後,若左群體[比基準值小的陣列]或右群體[比基準值大的陣列]的長度大於1的話,則再次呼叫quickSort函式(遞迴),最後將[比基準小的陣列] 、基準值、[比基準大的陣列]依序組合起來就是排序好的陣列。

https://ithelp.ithome.com.tw/upload/images/20210917/20128604WjIuStJJPG.png

用js實作快速排序法

const partition = (arr, left, right) => {
    let pivot = arr[right];
    //用來計算有幾個數字小於pivot
    let i = left - 1;
    //從最左邊找到pivot的前一個停止
    for (let j = left; j <= right - 1; j++) {
        if (arr[j] <= pivot) {
            i++;
            [arr[i], arr[j]] = [arr[j], arr[i]];
        }
    }
    [arr[i + 1], arr[right]] = [arr[right], arr[i + 1]];
    return i + 1;
};

const quickSort = (arr, left, right) => {
    if (left < right) {
        let sortedIndex = partition(arr, left, right);
        quickSort(arr, left, sortedIndex - 1);
        quickSort(arr, sortedIndex + 1, right);
    }
    return arr;
};

const arr = [2, 6, 3, 9, 1, 5];
quickSort(arr, 0, arr.length - 1);

//輸出 [1, 2, 3, 5, 6, 9]

第二種解法比較容易理解

函式內部實作為:

  • 先將陣列裡的最後一個元素定義為基準值 5
  • 接下來準備兩個陣列(比基準值大的陣列[A] 、比基準值小的陣列[B])
  • 跟基準值做比較, 比基準值大的值就放入A陣列 ,反之放入B陣列

若A or B陣列長度大於1,則再次呼叫quickSort函式(遞迴),最後將[比基準小的陣列] 、基準值、[比基準大的陣列]依序組合起來就是排序過後的陣列。
https://ithelp.ithome.com.tw/upload/images/20210917/20128604YmdF3xizWS.png

用js實作第二種快速排序法

const quickSort2 = (arr) => {
    if (arr.length <= 1) return arr;
    const small = [];
    const big = [];
    const pivot = arr[arr.length - 1];
    for (let i = 0; i < arr.length - 1; i++) {
        if (pivot > arr[i]) {
            small.push(arr[i]);
        } else {
            big.push(arr[i]);
        }
    }

    return [...quickSort2(small), pivot, ...quickSort2(big)];
};

const arr1 = [2, 6, 3, 9, 1, 5];
quickSort2(arr1);
//輸出 [1, 2, 3, 5, 6, 9]
  • 在最差的情況下,時間複雜度是O(n²)
  • 在最佳的情況下,時間複雜度是O(n log n)
  • 在平均情況下,時間複雜度為 O(n)*O(log n) = O(n log n)

上一篇
Day19:[排序演算法]Bubble Sort - 氣泡排序法
下一篇
Day21:[排序演算法]Heap Sort - 堆積排序法
系列文
每日攝取一點資料結構和演算法30
圖片
  直播研討會
圖片
{{ item.channelVendor }} {{ item.webinarstarted }} |
{{ formatDate(item.duration) }}
直播中

尚未有邦友留言

立即登入留言