成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

排序算法回顧(JavaScript)

jlanglang / 860人閱讀

摘要:回顧選擇排序,插入排序,冒泡排序,快速排序,希爾排序,歸并排序,堆排序以及如何計算時間復(fù)雜度學(xué)習(xí)文章同學(xué)的描述數(shù)據(jù)結(jié)構(gòu)等同學(xué)的十大經(jīng)典算法本文代碼也上傳到了排序算法回顧。但希爾排序是非穩(wěn)定排序算法。

回顧選擇排序,插入排序,冒泡排序,快速排序,希爾排序,歸并排序,堆排序以及如何計算時間復(fù)雜度
學(xué)習(xí)文章:hahda同學(xué)的javascript描述數(shù)據(jù)結(jié)構(gòu)、hustcc等同學(xué)的十大經(jīng)典算法

本文代碼也上傳到了 排序算法回顧(javascript)。

1.選擇排序

思路:從未排序的序列中選出最小(大)的元素,放進(jìn)已排好序的序列末尾。
時間復(fù)雜度:O(n^2)
算法穩(wěn)定性:不穩(wěn)定

// 定義一個函數(shù)用于交換
function swap (array, i, j) {
  let temp = array[i];
  array[i] = array[j];
  array[j] = temp;
}

function selectionSort (arr) {
  let minIndex;
  for (let i = 0; i < arr.length; i++) {
    minIndex = i;
    for (let j = i + 1; j < arr.length; j++) {  // 對未排序的序列進(jìn)行循環(huán),找出最小元素。
      if (arr[j] < arr[minIndex]) {
        minIndex = j;
      }
    }
    swap(arr, i, minIndex);                     // 最小元素與放如排好序的序列末尾。
  }
  return arr;
}

let arr = [1,2,8,4,3,6,10];

selectionSort(arr)   // 1,2,3,4,6,8,10

選擇排序所需要的元素比較次數(shù)為 (n-1) + (n-2) + ... + 1 = n*(n-1)/2 ,元素賦值次數(shù)界于 0 ~ 3(n-1) 之間,也就是原序列已排好序于原序列為反序兩種極端情況。

2.插入排序

思路:從第二個元素往后遍歷,從前面的序列中找到一個合適的位置進(jìn)行插入。
時間復(fù)雜度:O(n^2)
算法穩(wěn)定性:穩(wěn)定

let arr = [5,3,2,6,7,10,1];  // 進(jìn)行小到達(dá)排序

function InsertionSort(arr) {
  let len = arr.length;
  for (let i = 1; i < len; i++) {
    let curr = arr[i];                    // 要執(zhí)行插入操作的元素
    let j = i;                            // 從i開始往回遍歷
    while (j > 0 && arr[j-1] > curr) {   
// 不斷跟curr元素進(jìn)行比較,大于curr的往后退一位,最終給curr騰出一個插入的位置
      arr[j] = arr[j-1];
      j--;
    }
    arr[j] = curr                         // curr插入到合適的位置中
  }
  return arr;
}

console.log(InsertionSort(arr)); // 1,2,3,5,6,7,10

容易看出,當(dāng)序列已排好序的時候,元素比較的次數(shù)最少,比較次數(shù)為 n - 1 次,每一個元素只需要和前一個元素比較即可,當(dāng)序列是按反序排列,那么比較次數(shù)最多,比較次數(shù)為 n*(n-1)/2 。
元素賦值次數(shù)為等于比較次數(shù)加上 n - 1。

3.冒泡排序

思路:多次遍歷序列,比較相鄰元素,將最大(最小)元素像泡泡一樣冒到后面已排好序的序列中。
時間復(fù)雜度:O(n^2)
算法穩(wěn)定性:穩(wěn)定

function advanceBubbleSort1(arr){
    let len = arr.length;
    let flag;          // 設(shè)置一個標(biāo)記,如果某一輪沒有交換,表示已經(jīng)排好序了。不必再循環(huán)遍歷。
    for(let i = 1, i <= len - 1; i++){
        flag = false;
        for(let j = 0; j < len - i; j++){
            if(arr[j] > arr[j + 1]){
                temp = arr[j];
                arr[j] = arr[j + 1];
                arr[j + 1] = temp;
                flag = true;
            }
        }
        if(flag === false){
            break;
        }
    }
    return arr;
}
4.快速排序

快速排序是一個非常流行并且高效的排序算法。
它之所以高效是因為它在原位上進(jìn)行排序,不需要輔助的存儲空間。
思路:以最左元素作為主元進(jìn)行劃分,最后再將主元放回正確位置,遞歸。
平均時間復(fù)雜度 Θ(nlogn), 最壞的情況 θ(n^2)
算法穩(wěn)定性:不穩(wěn)定

在了解快速排序之前需要了解一個關(guān)鍵算法:劃分算法

function partition(arr, left ,right) {   // 分區(qū)操作
  var pivot = left,                      // 設(shè)定基準(zhǔn)值(pivot),即以最左元素為主元
      index = pivot + 1;
  for (var i = left + 1; i <= right; i++) {
      if (arr[i] < arr[pivot]) {
          swap(arr, i, index);
          index++;
      }        
  }
  swap(arr, pivot, index - 1);          // 最后把主元放回正確位置
  return index-1;
}


function swap(arr, i, j) {
  var temp = arr[i];
  arr[i] = arr[j];
  arr[j] = temp;
}

我們可以看到,整個劃分都在原數(shù)組上進(jìn)行,不需要引進(jìn)額外的輔助數(shù)組。
快速排序算法需要以劃分算法為核心:

function quickSort(arr, left, right) {
  var len = arr.length,
      partitionIndex;

  if (left < right) {
      partitionIndex = partition(arr, left, right);
      quickSort(arr, left, partitionIndex-1);
      quickSort(arr, partitionIndex+1, right);
  }
  return arr;
}

let arr = [1,6,3,8,5,0,7]

console.log(quickSort(arr, 0, 6))     // 0,1,3,5,6,7,8
5.希爾排序

希爾排序,也稱遞減增量排序算法,是插入排序的一種更高效的改進(jìn)版本。但希爾排序是非穩(wěn)定排序算法。
平均時間復(fù)雜度:O(nlogn)
算法穩(wěn)定性:不穩(wěn)定
思路:先將整個待排序的記錄序列分割成為若干子序列分別進(jìn)行直接插入排序,待整個序列中的記錄“基本有序”時,再對全體記錄進(jìn)行依次直接插入排序。

function shellSort(arr) {
  var len = arr.length,
      temp,
      gap = 1;
      while (gap < len/3) {
        gap = gap*3 + 1;
      }
      for (gap; gap > 0; gap = Math.floor(gap/3)) {
        for (var i = gap; i < len; i++) {
          temp = arr[i];
          for (var j = i-gap; i >= 0 && arr[j] > temp; j-=gap) {
            arr[j + gap] = arr[j];
          }
          arr[j + gap] = temp;
        }
      }
      return arr;
}

let arr = [10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0]

console.log(shellSort(arr)); // [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 ]
6.歸并排序

歸并排序采用分治法的思想。這里給出一種自上而下遞歸的方法。
思路:分半->分半->再分半->分到每組只剩下一個元素的時候就回溯
平均時間復(fù)雜度:O(nlogn)
算法穩(wěn)定性: 穩(wěn)定

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

function merge(left, right) {
  var result = [];
  while (left.length && right.length) {
    if (left[0] <= right[0]) {
      result.push(left.shift());
    } else {
      result.push(right.shift());
    }
  }
  while (left.length) result.push(left.shift());
  while (right.length) result.push(right.shift());

  return result;
}

let arr = [5, 3, 6, 8, 2, 0, 1];
console.log(mergeSort(arr)); // [ 0, 1, 2, 3, 5, 6, 8 ]
7.堆排序

堆排序是指利用堆這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計的一種排序算法。
大項堆:每個節(jié)點的值都大于或等于其子節(jié)點的值,用于升序排序
小項堆:每個節(jié)點的值都小于或等于其子節(jié)點,用于降序排序
平均時間復(fù)雜度:O(nlogn)
算法穩(wěn)定性: 不穩(wěn)定

var len;

function buildMaxHeap(arr) {   // 建立大頂堆
    len = arr.length;
    for (var i = Math.floor(len/2) - 1; i >= 0; i--) {
        heapify(arr, i);
    }
}

function heapify(arr, i) {     // 堆調(diào)整
    var left = 2 * i + 1,
        right = 2 * i + 2,
        largest = i;

    if (left < len && arr[left] > arr[largest]) {
        largest = left;
    }

    if (right < len && arr[right] > arr[largest]) {
        largest = right;
    }

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

function swap(arr, i, j) {
    var temp = arr[i];
    arr[i] = arr[j];
    arr[j] = temp;
}

function heapSort(arr) {
    buildMaxHeap(arr);

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

let arr = [4,3,8,10,11,13,7,30,17,26];
console.log(heapSort(arr))  // [ 3, 4, 7, 8, 10, 11, 13, 17, 26, 30 ]
8.如何估算時間復(fù)雜度

了解幾個概念:

O 符號表示一個運行時間的上界。

Ω 符號表示一個運行時間的下界。

θ 符號表示一個精準(zhǔn)描述。

可以這樣幫助理解,O 類似于 <= ,Ω 類似于 >=, θ 類似于 = ,但只能說是類似于。

(1) 計算迭代次數(shù)

let i = 0;
for (let i = 0; i < n; i++) {
  i ++;
}

可以看到迭代次數(shù)為n,所以時間復(fù)雜度為 θ(n)

(2) 計算基本運算的頻度

什么是基本運算呢?

在分析搜索和排序算法時,如果比較是元運算(不能再細(xì)化的運算),可以選擇它為基本運算

矩陣乘法算法中,可以選擇數(shù)量乘法運算

遍歷鏈表時,可以選擇設(shè)置或更新指針的運算

再圖的遍歷中可以選擇訪問結(jié)點的動作和被訪問結(jié)點的計算

如上一份代碼

let i = 0;
for (let i = 0; i < n; i++) {
  i ++;
}

選擇自加運算,同理得時間復(fù)雜度 θ(n)

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/93497.html

相關(guān)文章

  • Conflux & TokenGazer AMA活動內(nèi)容回顧

    摘要:安全性不可更改性排序結(jié)果不能被壞人的攻擊更改。這也是很嚴(yán)重的公鏈安全事故??偠灾?,通過設(shè)計安全的拓?fù)渑判蛩惴?,解決交易順序問題。區(qū)塊排序的一致可以保證無效交易標(biāo)記的一致。樞軸鏈和分叉鏈的區(qū)塊獎勵計算規(guī)則是一致的。 showImg(https://segmentfault.com/img/remote/1460000017710155?w=893&h=380); 12月27日,Conf...

    littlelightss 評論0 收藏0
  • Conflux & TokenGazer AMA活動內(nèi)容回顧

    摘要:安全性不可更改性排序結(jié)果不能被壞人的攻擊更改。這也是很嚴(yán)重的公鏈安全事故??偠灾ㄟ^設(shè)計安全的拓?fù)渑判蛩惴?,解決交易順序問題。區(qū)塊排序的一致可以保證無效交易標(biāo)記的一致。樞軸鏈和分叉鏈的區(qū)塊獎勵計算規(guī)則是一致的。 showImg(https://segmentfault.com/img/remote/1460000017710155?w=893&h=380); 12月27日,Conf...

    DesGemini 評論0 收藏0
  • Conflux & TokenGazer AMA活動內(nèi)容回顧

    摘要:安全性不可更改性排序結(jié)果不能被壞人的攻擊更改。這也是很嚴(yán)重的公鏈安全事故??偠灾ㄟ^設(shè)計安全的拓?fù)渑判蛩惴?,解決交易順序問題。區(qū)塊排序的一致可以保證無效交易標(biāo)記的一致。樞軸鏈和分叉鏈的區(qū)塊獎勵計算規(guī)則是一致的。 showImg(https://segmentfault.com/img/remote/1460000017710155?w=893&h=380); 12月27日,Conf...

    smallStone 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<