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

来源: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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...