Sorting 排序
排序 Sorting: Comparison Sort
氣泡排序 Bubble Sort - Time O(n) or O(n^2) / Space - O(1)
選擇排序 Selection Sort - Time O(n^2) / Space - O(1)
插入排序 Insertion Sort - Time O(n) or O(n^2) / Space - O(1)
合併排序 Merge Sort - Time O(n log n) / Space - O(n)
快速排序 Quick Sort - Time O(n log n) / Space - O(log n)
Non-Comparison Sort
Array.sort()
JavaScript中的Array.sort()方法使用的是 取決於瀏覽器 Chrome V8 主要用快速排序 QuickSort 也有一些Merge Sort 和 Insertion Sortsort()
會將元素轉換成 字串 來排序,預設排序為unicode 從 a 排到 z , 從 小 排到 大,若有十位數以上數字則會產生排序問題,因此需要用以下方法來解決。
1 2 3 4 5 6 7 8 9 10 let ary = [100 , 1 , 4 , 50 ]; ary.sort (); console .log (ary); ary.sort ((a,b ) => b - a); console .log (ary); ary.sort ((a,b ) => a - b); console .log (ary);
氣泡排序 Bubble Sort Time - O(n) best or O(n^2) for average and worst Space - O(1)
第一個與第二個比較大小是否交換順序,不用交換或交換完後跟下一個比大小是否交換順序,以此類推,整排結束後,再接下從第二個開始與第三個比,以此類推….最後完成排序
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 const numbersArr = [99 , 44 , 6 , 2 , 1 , 5 , 63 , 87 , 283 , 4 , 0 ];function bubbleSortAscending (array ) { for (let i = 0 ; i < array.length ; i++){ for (let j = 0 ; j < array.length ; j++){ if (array[j] > array[j + 1 ]){ let temp = array[j]; array[j] = array[j + 1 ]; array[j + 1 ] = temp; } } } return array; } function bubbleSortDescending (array ) { for (let i = 0 ; i < array.length ; i++){ for (let j = 0 ; j < array.length ; j++){ if (array[j] < array[j + 1 ]){ let temp = array[j]; array[j] = array[j + 1 ]; array[j + 1 ] = temp; } } } return array; } console .log (bubbleSortAscending (numbersArr));console .log (bubbleSortDescending (numbersArr));
選擇排序 Selection Sort Time - O(n^2) Space - O(1)
loop整個array去找出誰是最小值(最大值),然後把它放到第一個,接著從第二個開始再loop找出誰是最小值(最大值)再放到第一個,以此類推….最後完成排序
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 const numbers = [99 , 44 , 6 , 2 , 1 , 5 , 63 , 87 , 283 , 4 , 0 ];function selectionSortAscending (array ) { for (let i = 0 ; i < array.length ; i++) { let minIndex = i; let temp = array[i]; for (let j = i + 1 ; j < array.length ; j++) { if (array[j] < array[minIndex]) { minIndex = j; } } array[i] = array[minIndex]; array[minIndex] = temp; } return array; } function selectionSortDescending (array ) { for (let i = 0 ; i < array.length ; i++) { let maxIndex = i; let temp = array[i]; for (let j = i + 1 ; j < array.length ; j++) { if (array[j] > array[maxIndex]) { maxIndex = j; } } array[i] = array[maxIndex]; array[maxIndex] = temp; } return array; } console .log (selectionSortAscending (numbers));console .log (selectionSortDescending (numbers));
插入排序 Insertion Sort Time - O(n)(for already sorted array) O(n^2) for average and worst Space - O(1)
最熟悉的例子就是玩接龍時,要先把手上的牌依照大小排列的插入法 先從第二個與前一個比較,若較大或相等則不動,若較小則交換到第一個,接下來換第三個數字與前一個比較,若較大或相等則不動,若較小則交換到第一個,再與前一個比較,以此類推到第一個…再以此類推直到最後一個比較完
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 const numbers = [99 , 44 , 6 , 2 , 1 , 5 , 63 , 87 , 283 , 4 , 0 ];function sort (array ) { for (let i = 1 ; i < array.length ; i++) { let current = array[i]; for (let j = i - 1 ; j >= 0 ; j--) { if (current >= array[j]) { break ; } else { array[j + 1 ] = array[j] array[j] = current; } } } return array; } console .log (insertionSort (numbers));
合併排序 Merge Sort Time O(n log n) Space - O(n)
把 數列陣列 拆開再拆開 到最後剩一下一個數值,然後兩個陣列各自從第一個數值比較,小的拉出放前,再來比較拿出後的下一個,小的拉出放前到陣列比較結束,合併成一個新陣列,再依序必較其他合併的陣列到最後合併成一個陣列,排序完成。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 const numbers = [99 , 44 , 1 , 2 , 10 ];function mergeSort (array ) { if (array.length === 1 ) { return array } let middle = Math .floor (array.length / 2 ); let left = array.slice (0 , middle); let right = array.slice (middle); return merge (mergeSort (left), mergeSort (right)); } function merge (left, right ) { console .log ('left' , left, 'right' , right); const result = []; let leftIndex = 0 ; let rightIndex = 0 ; while (leftIndex < left.length && rightIndex < right.length ){ if (left[leftIndex] < right[rightIndex]){ result.push (left[leftIndex]); leftIndex++ }else { result.push (right[rightIndex]); rightIndex ++ } } return [...result, ...left.slice (leftIndex), ...right.slice (rightIndex)]; } console .log (mergeSort (numbers));
合併排序題目:輸出逆序數對的總數 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 function countInvertedPairs (array ){ let count = 0 ; function mergeSort (arr ){ if (arr.length <= 1 ){ return arr; } let middle = Math .floor (arr.length / 2 ); let left = arr.slice (0 , middle); let right = arr.slice (middle); return merge (mergeSort (left), mergeSort (right)); } function merge (left, right ){ let i = 0 ; let j = 0 ; let result = []; while (i < left.length && j < right.length ){ if (left[i]<= right[j]){ result.push (left[i]); i++; }else { result.push (right[j]); count += left.length - i; j++; } } return [...result, ...left.slice (i), ...right.slice (j)]; } mergeSort (array); return count; }
快速排序 Quick Sort Time - O(n log n) Space - O(log n)
先隨機選一個數作為pivot 基準數,最左邊有一個pointer L,最右邊也有一個pointer R 跟pivot做比較,比pivot小的放在
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 const numbers = [8 , 9 , 2 , 5 , 1 ]; function quickSort (arr, left, right ) { if (left < right) { let partitionIndex = partition (arr, left, right); quickSort (arr, left, partitionIndex - 1 ); quickSort (arr, partitionIndex + 1 , right); } return arr; } function partition (arr, left, right ) { let pivot = right let partitionIndex = left; for (let i = left; i < right; i++) { if (arr[i] < arr[pivot]) { swap (arr, i, partitionIndex); partitionIndex++; } } swap (arr, right, partitionIndex); return partitionIndex; } function swap (arr, index1, index2 ) { [arr[index1], arr[index2]] = [arr[index2], arr[index1]]; } console .log (quickSort (numbers, 0 , numbers.length - 1 ));
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 const numbers = [8 , 9 , 2 , 5 , 1 ]; function quickSort (arr ) { if (arr.length <= 1 ) return arr; const [pivot, ...ary] = arr; const left = []; const right = []; ary.forEach (item => { if (item < pivot) { left.push (item); } else { right.push (item); } }) return [...quickSort (left), pivot, ...quickSort (right)] } console .log (quickSort (numbers));
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 const numbers = [8 , 9 , 2 , 5 , 1 ];function quickSort (array, left, right ) { if (array.length <= 1 ) { return array; } let indexCounter = 0 ; for (let i = 0 ; i < array.length ; i++) { if (array[i - indexCounter] > array[right - indexCounter]) { array.push (array.splice (i - indexCounter, 1 )[0 ]); indexCounter++; } } let leftArray = array.slice (left, right - indexCounter); let rightArray = array.slice (right - indexCounter + 1 , array.length ); return [...quickSort (leftArray, 0 , leftArray.length - 1 ), array[right - indexCounter], ...quickSort (rightArray, 0 , rightArray.length - 1 )]; } console .log (quickSort (numbers, 0 , numbers.length - 1 ));
Counting Sort 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 const numbers = [10 , 2 , 4 , 5 , 1 , 2 , 1 , 9 ];function CountingSort (array ) { let result = []; let collection = [...Array (10 ).fill (0 )]; array.forEach (item => { collection[item - 1 ]++ }); collection.forEach ((item, i ) => { for (let j = 1 ; j <= item; j++){ result.push (i + 1 ); } }) return result; } console .log (CountingSort (numbers));
Interview Question
Sort 10 schools around your house by distance: insertion sort
eBay sorts listings by the current Bid amount: radix or counting sort
Sport scores on ESPN quick sort
Massive database (can’t fit all into memory) needs to sort through past year’s user data merge sort
Almost sorted Udemy review data needs to update and add 2 new reviews insertion sort
Temperature Records for the past 50 years in Canada radix or counting sort quick sort if decimal places
Large user name database needs to be sorted. Data is very random. quick sort
You want to teach sorting for the first time bubble sort
學習與參考資料