前端學演算法之演算法複雜度

来源:https://www.cnblogs.com/xiaohuochai/archive/2018/01/05/8203717.html
-Advertisement-
Play Games

前面的話 本文將詳細介紹演算法複雜度 大O表示法 大O表示法是描述演算法的性能和複雜程度。 分析演算法時,時常遇到以下幾類函數 如何衡量演算法的效率?通常是用資源,例如CPU(時間)占用、記憶體占用、硬碟占用和網路占用。當討論大O表示法時,一般考慮的是CPU(時間)占用 下麵用一些例子來理解大O表示法的規則 ...


前面的話

  本文將詳細介紹演算法複雜度

 

大O表示法

  大O表示法是描述演算法的性能和複雜程度。 分析演算法時,時常遇到以下幾類函數

符號             名稱
O(1)            常數的
O(log(n))        對數的
O((log(n))c)    對數多項式的
O(n)            線性的
O(n2)            二次的
O(nc)            多項式的
O(cn)            指數的

  如何衡量演算法的效率?通常是用資源,例如CPU(時間)占用、記憶體占用、硬碟占用和網路占用。當討論大O表示法時,一般考慮的是CPU(時間)占用

  下麵用一些例子來理解大O表示法的規則

【O(1)】

function increment(num){ 
  return ++num;
}

  假設運行increment(1)函數,執行時間等於X。如果再用不同的參數(例如2)運行一次increment函數,執行時間依然是X。和參數無關,increment函數的性能都一樣。因此,我們說上述函數的複雜度是O(1)(常數)

【O(n)】

  現在以順序搜索演算法為例:

function sequentialSearch(array, item){ 
  for (var i=0; i<array.length; i++){
    if (item === array[i]){ //{1} 
      return i;
    }
  }
  return -1;
}

  如果將含10個元素的數組([1, ..., 10])傳遞給該函數,假如搜索1這個元素,那麼,第一次判斷時就能找到想要搜索的元素。在這裡我們假設每執行一次行{1} ,開銷是1。

  現在,假如要搜索元素11。行{1}會執行10次(遍曆數組中所有的值,並且找不到要搜索的元素,因而結果返回 -1)。如果行{1}的開銷是1,那麼它執行10次的開銷就是10,10倍於第一種假設

  現在,假如該數組有1000個元素([1, ..., 1000])。搜索1001的結果是行{1}執行了1000次(然後返回-1)

  sequentialSearch函數執行的總開銷取決於數組元素的個數(數組大小),而且也和搜索的值有關。如果是查找數組中存在的值,行{1}會執行幾次呢?如果查找的是數組中不存在的值,那麼行{1}就會執行和數組大小一樣多次,這就是通常所說的最壞情況

  最壞情況下,如果數組大小是10,開銷就是10;如果數組大小是1000,開銷就是1000。可以得出sequentialSearch函數的時間複雜度是O(n),n是(輸入)數組的大小

  回到之前的例子,修改一下演算法的實現,使之計算開銷:

function sequentialSearch(array, item){
 var cost = 0;
 for (var i=0; i<array.length; i++){
  cost++;
  if (item === array[i]){ //{1}
    return i;
  }
 }
 console.log('cost for sequentialSearch with input size ' +  array.length + ' is ' + cost);
 return -1;
} 

  用不同大小的輸入數組執行以上演算法,可以看到不同的輸出

O(n2)】

  用冒泡排序做O(n2)的例子:

function swap(array, index1, index2){
 var aux = array[index1];
 array[index1] = array[index2];
 array[index2] = aux;
}
function bubbleSort(array){
 var length = array.length;
 for (var i=0; i<length; i++){ //{1}
  for (var j=0; j<length-1; j++ ){ //{2}
    if (array[j] > array[j+1]){
      swap(array, j, j+1);
    }
  }
 }
} 

  假設行{1}和行{2}的開銷分別是1。修改演算法的實現使之計算開銷:

function bubbleSort(array){
 var length = array.length;
 var cost = 0;
 for (var i=0; i<length; i++){ //{1}
  cost++;
  for (var j=0; j<length-1; j++ ){ //{2}
    cost++;
    if (array[j] > array[j+1]){
      swap(array, j, j+1);
    }
  }
 }
 console.log('cost for bubbleSort with input size ' + length + ' is ' + cost);
} 

  如果用大小為10的數組執行bubbleSort,開銷是100(102)。如果用大小為100的數組執 行bubbleSort,開銷就是10 000(1002)。需要註意,我們每次增加輸入的大小,執行都會越來越久

  時間複雜度O(n)的代碼只有一層迴圈,而O(n2)的代碼有雙層嵌套迴圈。如 果演算法有三層遍曆數組的嵌套迴圈,它的時間複雜度很可能就是O(n3)

 

時間複雜度

  下圖比較了前述各個大O符號表示的時間複雜度:

arithmetic21

  下表是常用數據結構的時間複雜度

arithmetic22

  下表是圖的時間複雜度: 

arithmetic23

  下表是排序演算法的時間複雜度: 

arithmetic24

  下表是搜索演算法的時間複雜度: 

arithmetic25

 

NP

  一般來說,如果一個演算法的複雜度為O(nk),其中k是常數,我們就認為這個演算法是高效的,這就是多項式演算法

  對於給定的問題,如果存在多項式演算法,則計為P(polynomial,多項式)

  還有一類NP(nondeterministicpolynomial,非確定性多項式)演算法。如果一個問題可以在多項式時間內驗證解是否正確,則計為NP。如果一個問題存在多項式演算法,自然可以在多項式時間內驗證其解。因此,所有的P都是NP。然而,P=NP是否成立,仍然不得而知。NP問題中最難的是NP完全問題,它滿足以下兩個條件:(1)是NP問題,也就是說,可以在多項式時間內驗證解,但還沒有找到多項式演算法;(2)所有的NP問題都能在多項式時間內歸約為它。為了理解問題的歸約,考慮兩個決策問題L和M。假設演算法A可以解決問題L,演算法B可以驗證輸入y是否為M的解。目標是找到一個把L轉化為M的方法,使得演算法B可以用於構造演算法A

  還有一類問題,只需滿足NP完全問題的第二個條件,稱為NP困難問題。因此,NP完全問題也是NP困難問題的子集

  下麵是滿足P < > NP時,P、NP、NP完全和NP困難問題的歐拉圖: 

arithmetic26

 

  非NP完全的NP困難問題的例子有停機問題和布爾可滿足性問題(SAT)。 NP完全問題的例子有子集和問題、旅行商問題、頂點覆蓋問題等等

  我們提到的有些問題是不可解的。然而,仍然有辦法在符合要求的時間內找到一個近似解。啟髮式演算法就是其中之一。啟髮式演算法得到的未必是最優解,但足夠解決問題了。啟髮式演算法的例子有局部搜索、遺傳演算法、啟髮式導航、機器學習等

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • javascript這門語言一直就像一位帶著面紗的美女,總是看不清,摸不透,一直專註伺服器端,也從來沒有特別重視過,直到最近幾年,javascript越來越重要,越來越通用。最近和前端走的比較近,藉此機會,好好鞏固一下相關知識點。 1.初識replace 在js中有兩個replace函數 一個是lo ...
  • javaScript調試工具console命令的使用 我最先認識到console命令是在javaScript中看到的,當時只是知道它的console.log()命令的使用,並沒有深究。後來,特意去查了下,並通過這篇博客記錄下來。 一、console是幹嘛的? 我的理解是: 在瀏覽器控制臺中顯示信息, ...
  • jQuery是一個輕量級的JavaScript庫,裡面包含所有的jQuery方法。 如果想要使用這些方法,那麼必須首先引用這個庫。 先看一段代碼: 使用$()方法獲取指定元素,然後利用hide()方法隱藏元素。要想使用這些方法,比如首先引入jQuery庫。 上面代碼引入庫的方式是: 從網路上引入jQ ...
  • 一.CSS語法: CSS語法規則由兩個部分組成: (1).選擇器:它的作用是用來匹配要應用css代碼的元素。 (2).Rules:屬性聲明語句,用來具體控制元素的表現,屬性和屬性值之間要用冒號分割。如果有多條聲明語句,那麼相互之間用分號分隔,最後一條語句後面可以省略分號。 二代碼實例: <!doct ...
  • 在HTML中,共有6個級別的標簽:<h1>~<h6>。 標題數字越小,字體就會越大,標題的級別也就越高。 標題標簽的使用對於搜索引擎優化也有著比較重要的作用,這裡就不具體介紹了。 代碼實例: ...
  • 表達式是JavaScript中的短語,那麼語句就是JavaScript的整句或者命令。 JavaScript語句是以分號結尾的(分號有時候是可以省略的,需要保持語義完整性)。 如果說表達式是人體的細胞或者軟組織的話,那麼語句就是更高層次的人體器官,它能夠完成一些較為複雜的操作,改變程式的運行狀態。復 ...
  • 通過查詢瞭解到博客園是有開發博客查詢相關的介面的,列表如下: 但是我們打開其中一個介面的話會發現提供的介面返回的是xml格式的內容,因此如果需要後臺轉發為前臺需要的格式還需要把xml轉換為json數據: 那麼我們一步一步來,首先需要Node將這個介面代理轉發為自己的介面,其實只需要express的r ...
  • 效果: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...