JavaScript中的多種排序算法

桶排序

  • 性能:浪費空間,且只能比較天然數,時間複雜度是 O(m+n)
  • 須要對數據範圍在0~n之間的整數進行排序
var testArr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function skipSort (n, skipArr) {
  let arr = []
  let sortArr = []
  arr.length = n
  for (let i = 0; i <= n; i++) {
    arr[i] = 0
  }
  for (let j = 0; j < skipArr.length; j++) {
    arr[skipArr[j]]++
  }
  for (let t = 0; t <= n; t++) {
    if (arr[t]) {
      for (let k = 1; k <= arr[t]; k++) {
        sortArr.push(t)
      }
    }
  }
  return sortArr
}

console.log('桶排序:', skipSort(10, testArr))
複製代碼

冒泡排序

  • 性能:時間複雜度是O(n^2),執行效率低
  • 元素項向上移動至正確的順序,就好像氣泡往上冒同樣
  • 從小到大排序,比較兩個相鄰的項,若是第一個大於第二個則交換他們的位置
var arr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function bubbleSort (arr) {
  let len = arr.length
  for (let i = 0; i < len - 1; i++) {
    for (let j = 0; j < len - 1 - i; j++) {
        if (arr[j] > arr[j + 1]) {
            let temp = arr[j]
            arr[j] = arr[j + 1]
            arr[j + 1] = temp
        }
    }
  }
  return arr
}
console.log('冒泡排序:',bubbleSort(arr))
複製代碼

快速排序

  • chrome使用快速排序的一個變體做爲Array.prototype.sort的實現
  • 性能:時間複雜度最差是O(n^2),平均時間複雜度爲O(NlogN)
  • 首先,在數組中選擇一箇中間項做爲主元
  • 建立兩個指針,左邊的指向數組第一個項,右邊的指向最後一個項。移動右指針,直到找到一個比主元小的項,接着,移動左邊的指針,直到找到一個比主元大的項,而後交換它們。重複這個過程,直到左側的指針超過了右側的指針。這個使比主元小的都在左側,比主元大的都在右側。這一步叫劃分操做
  • 接着,算法對劃分後的小數組(較主元小的值組成的的小數組, 以及較主元大的值組成的小數組)重複以前的兩個步驟,直到排序完成
  • 切記:必定要先從右往左找,再從左往右找,不然會出錯
var arr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function quickSort (arr, left, right) {
  let len = arr.length
  left = typeof left === 'undefined' ? 0 : left
  right = typeof right === 'undefined' ? len - 1 : right
  if (left > right) {
    return
  }
  let temp = arr[left]
  let i = left
  let j = right
  let t
  while (i !== j) {
    while (arr[j] >= temp && i < j) {
      j--
    }
    while (arr[i] <= temp && i < j) {
      i++
    }
    if (i < j) {
      t = arr[i]
      arr[i] = arr[j]
      arr[j] = t
    }
  }
  arr[left] = arr[i]
  arr[i] = temp
  quickSort(arr, left, i - 1)
  quickSort(arr, i + 1, right)
  return arr
}
console.log('快速排序:',quickSort(arr))
複製代碼

選擇排序

  • 大概思路是找到最小的放在第一位,找到第二小的放在第二位,以此類推 算法複雜度O(n^2)
var arr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function selectSort (arr) {
  let len = arr.length
  let minIndex
  let temp
  for (let i = 0; i < len; i++) {
    minIndex = i
    temp = arr[i]
    for (let j = i + 1; j < len; j++) {
        if (arr[j] < arr[minIndex]) { 
            minIndex = j
        }
    }
    arr[i] = arr[minIndex]
    arr[minIndex] = temp
  }
  return arr
}
console.log('選擇排序:',selectSort(arr))
複製代碼

插入排序

  • 每次排一個數組項,假設數組的第一項已經排序
  • 接着,把第二項與第一項進行對比,第二項是該插入到第一項以前仍是以後
  • 第三項是該插入到第一項以前仍是第一項以後仍是第三項
var arr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function insertSort (arr) {
  let len = arr.length
  let preIndex
  for (let i = 1; i < len; i++) {
    preIndex = i - 1
    current = arr[i]
    while (preIndex >= 0 && arr[preIndex] > current) {
      arr[preIndex + 1] = arr[preIndex]
      preIndex--
    }
    arr[preIndex + 1] = current
  }
  return arr
}
console.log('插入排序:', insertSort(arr))
複製代碼

歸併排序

  • Mozilla Firefox使用歸併排序做爲Array.prototype.sort的實現
  • 歸併排序是一種分治算法。本質上就是把一個原始數組切分紅較小的數組,直到每一個小數組只有一個位置,接着把小數組歸併成較大的數組,在歸併過程當中也會完成排序,直到最後只有一個排序完畢的大數組
  • 分治算法的基本思想是將一個規模爲N的問題分解爲K個規模較小的子問題,這些子問題相互獨立且與原問題性質相同。求出子問題的解,就可獲得原問題的解。
var arr = [3, 5, 3, 5, 9, 7, 6]
console.log('原數組:', arr)

function mergeSort (arr) {
  let len = arr.length
  if (len < 2) {
    return arr
  }
  let middle = Math.floor(len / 2)
  let left = arr.slice(0, middle)
  let right = arr.slice(middle)
  return merge(mergeSort(left), mergeSort(right))
}

function merge (left, right) {
  let result = []
  while (left.length && right.length) {
    if (left[0] < right[0]) {
      result.push(left.shift())
    } else {
      result.push(right.shift())
    }
  }
  result.push(...left)
  result.push(...right)
  return result
}
console.log('歸併排序:', mergeSort(arr))
複製代碼

堆排序

  • 堆排序其實是利用堆的性質來進行排序的,咱們一般說的堆就是二叉堆,二叉堆又稱徹底二叉樹或者近似徹底二叉樹堆排序是選擇排序的一種。能夠利用數組的特色快速定位指定索引的元素。數組能夠根據索引直接獲取元素,時間複雜度爲O(1)算法

  • 最大堆的特性以下:chrome

    • 父結點的鍵值老是大於或者等於任何一個子節點的鍵值
    • 每一個結點的左子樹和右子樹都是一個最大堆
  • 最小堆的特性以下:api

    • 父結點的鍵值老是小於或者等於任何一個子節點的鍵值
    • 每一個結點的左子樹和右子樹都是一個最小堆
  • 算法思想數組

    將待排序序列構形成一個大頂堆,此時,整個序列的最大值就是堆頂的根節點。將其與末尾元素進行交換,此時末尾就爲最大值。而後將剩餘n-1個元素從新構形成一個堆,這樣會獲得n個元素的次小值。如此反覆執行,便能獲得一個有序序列了。性能

    • 構建初始堆。通常升序採用大頂堆,降序採用小頂堆。從第一個非葉子結點從下至上,從右至左調整結構;
    • 交換堆頂元素和末尾元素,使最大值沉到數組末尾,從新調整堆結構,使其知足定義;
    • 而後繼續交換堆頂元素與當前末尾元素,反覆執行調整+交換,直到整個序列有序。
const arr = [21, 34, 56, 2, 4, 9, 87]
console.log('原數組:', arr)

let len = arr.length

// 構建最大堆
function buildMaxHeap (arr) {
  let mid = Math.floor(len / 2)
  for (let i = mid; i >= 0; i--) {
    heapify(arr, i)
  }
}

// 子數的調整
function heapify (arr, i) {
  let maxIndex = i
  let left = 2 * i + 1
  let right = 2 * i + 2

  if (left <len && arr[maxIndex] < arr[left]) {
    maxIndex = left
  }

  if (right <len && arr[maxIndex] < arr[right]) {
    maxIndex = right
  }

  if (maxIndex !== i) {
    swap(arr, i, maxIndex)
    heapify(arr, maxIndex)
  }
}

// 交換兩值
function swap (arr, i, j) {
  [arr[i], arr[j]] = [arr[j], arr[i]]
}

// 堆排序
function heapSort (arr) {
  buildMaxHeap(arr)

  for (let i = arr.length - 1; i>=0; i--) {
    swap(arr, 0, i)
    len--
    heapify(arr, 0)
  }

  return arr
}

console.log('堆排序:', heapSort(arr))
複製代碼

各個排序算法的性能比較

相關文章
相關標籤/搜索