演算法學習-1 演算法複雜度

来源:https://www.cnblogs.com/wwwen/archive/2022/11/18/16902006.html
-Advertisement-
Play Games

一 演算法複雜度 演算法複雜度分為時間複雜度和空間複雜度。時間複雜度是指執行演算法所需要的計算工作量;而空間複雜度是指執行這個演算法所需要的記憶體空間。 演算法的複雜性體運行該演算法時的電腦所需資源的多少,電腦資源最重要的是時間和空間(即寄存器)資源,因此複雜度分為時間和空間複雜度。 二 時間複雜度 2.1 ...


一 演算法複雜度

演算法複雜度分為時間複雜度和空間複雜度。時間複雜度是指執行演算法所需要的計算工作量;而空間複雜度是指執行這個演算法所需要的記憶體空間。

演算法的複雜性體運行該演算法時的電腦所需資源的多少,電腦資源最重要的是時間和空間(即寄存器)資源,因此複雜度分為時間和空間複雜度。

 

二 時間複雜度

2.1 關於時間複雜度

一個演算法花費的時間與演算法中語句的執行次數成正比例,演算法中語句執行次數越多,它花費時間就越多。一個演算法中的語句執行次數稱為語句時間頻度。記為T(n)。

 

假設演算法的問題規模為n,演算法中操作單元的數量用函數 f(n) 來表示,當n趨近於無窮大時,T(n) / f(n) 的極限值為不等於零的常數,

即隨著數據規模n的增大,演算法執行時間的增長率和 f(n)  的增長率相同,則稱f(n)是T(n)的同數量級函數。記作T(n)=O(f(n))。

則稱O(f(n)) 為演算法的漸進時間複雜度,簡稱時間複雜度(Time complexity)。

 

時間複雜度用大O符號(big O)表述,不包括f(n)函數的低階項和首項繫數。

不同的數據規模n可能造成演算法的運行時間不同,因此通常使用演算法的最壞情況來估計時間複雜度。

2.2 常數操作

常數時間的操作:一個操作如果和樣本的數據量沒有關係,每次都是固定時間內完成操作,叫做常數操作。

常數時間的操作是指演算法代碼中的指令都是固定時間的操作,指令是和數據量沒有關係,比如加、減、乘、除、模、位運算,或數組的定址。

2.3 常數時間

若對於一個演算法,T(n)的上界與輸入n的大小無關,則稱其具有常數時間,記作O(1)時間。例如訪問數組中的單個元素,是常數因為訪問它只需要一條指令。

但是,找到無序數組中的最小元素則不是,因為這需要遍歷所有元素來找出最小值。這是一項線性時間的操作,也稱O(n)時間。

int i = 1;
int j = 2;
i = j++;
j = j << 2;
int m = (i + j) / 2;

2.4 線性時間

如果一個演算法的時間複雜度為O(n),則稱這個演算法具有線性時間,或O(n)時間。這意味著對於足夠大的輸入,運行時間增加的大小與輸入成線性關係。

例如,一個計算列表所有元素的和的程式,需要的時間與列表的長度成正比。

for (int i = 1; i <= n; ++i)
{
    int j = i;
}

2.5 對數時間

若演算法的T(n) = O(log n),則稱其具有對數時間。由於電腦使用二進位的記數系統,未寫明底數時,預設以2為底。

常見的具有對數時間的演算法有二叉樹的相關操作和二分查找。

對數時間的演算法是非常有效的,因為每增加一個輸入,其所需要的額外計算時間會變小。

int i = 1;
while (i < n)
{
    i = i * 2;
}

2.6 線性對數時間

若演算法時間複雜度T(n) = O(nlog n),則稱這個演算法具有線性對數時間。線性對數時間增長得比線性時間要快,但是對於任何含有n,且n的冪指數大於1的多項式時間來說,線性對數時間增長得慢。

例如方法Method1

void Method1(int n)
{
    for (int i = 0; i < n; i++)
    {
        Method2(n);
    }
}

void Method2(int n)
{
    for (int j = 1; j <= n; j = j * 2)
    {
        Console.WriteLine(j);
    }
}

 

二 空間複雜度

空間複雜度(Space Complexity):是對一個演算法在運行過程中臨時占用存儲空間大小的量度,記做S(n)=O(f(n))。

比如插入的時間複雜度是O(n^2),空間複雜度是O(1) 。而一般的遞歸演算法的空間複雜度為O(n),因為每次遞歸都要存儲返回信息。

例,空間複雜度O(1)

int i = 1;
int j = 2;
i = j++;
j = j << 2;
int m = (i + j) / 2;

空間複雜度O(n)

int[] m = new int[n];
int j = 0;
for (i = 1; i <= n; ++i)
{
    j = i;
    j++;
}

 

以上。


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

-Advertisement-
Play Games
更多相關文章
  • 基於 SpringWeb(5.3.23)的介面請求分析 前情提要 假定當前 Web 項目中有如下實體類和介面: package com.example.entity; public class WebUser { private String name; private Integer age; p ...
  • 創建第一個springmvc程式 1、創建父項目文件,導入依賴,刪除src文件夾 pom.xml文件 <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</ ...
  • 算術運算符 +(加) -(減) *(乘) /(除) %(取餘) ++(自增) --(自減) 註意:/(除):兩個整數相除,其結果一定是整數,小數位電腦自動略去 例: int num1 = 15; int num2 = 4; 1. int result = num1/num2; system.out ...
  • 目錄 一.簡介 1.freeglut 2.glew 3.glut 4.glfw 5.glad 二.分類 1.視窗管理 2.函數載入 三.組合使用 1.freeglut + glew 2.glfw + glew 3.glfw + glad 四.猜你喜歡 零基礎 OpenGL ES 學習路線推薦 : O ...
  • 繼承: 強調類與類之間的關係 組合: 強調對象和對象之間的關係 清楚python支持多繼承,從而涉及到一些MRO的點,這裡不做贅述,在實際工作過程中,我們經常會使用繼承來實現代碼復用,如果僅僅是為了復用,還是比較推薦使用組合方式,因為繼承方式,使得類與類之間的耦合性變得異常緊密,這多少違背了迪米特法 ...
  • 故事背景 最近同事遇到一個比較奇怪的問題,直接開門見山吧。在動態庫中調用靜態庫直接報錯了recompile with -fPIC,查看cmake的寫法也沒有問題,而且也是第一次遇見這個問題,所以就開啟了我的好奇之路。 探索之路 說實話我不喜歡百度,因為千篇一律,你抄我的我抄你的,沒有任何參考價值,直 ...
  • 一、序言 在日常一線開發過程中,總有列表轉樹的需求,幾乎是項目的標配,比方說做多級菜單、多級目錄、多級分類等,有沒有一種通用且跨項目的解決方式呢?幫助廣大技術朋友給業務瘦身,提高開發效率。 本文將基於Java8的Lambda 表達式和Stream等知識,使用TreeUtils工具類實現一行代碼完成列 ...
  • 面試官: 小伙子,我看你簡歷上寫的項目中用到了線程池,你知道線程池是怎樣實現復用線程的? 這面試官是不是想坑我?是不是擺明瞭不讓我通過? 難道你不應該問線程池有哪些核心參數?每個參數具體作用是什麼? ...
一周排行
    -Advertisement-
    Play Games
  • 1、預覽地址:http://139.155.137.144:9012 2、qq群:801913255 一、前言 隨著網路的發展,企業對於信息系統數據的保密工作愈發重視,不同身份、角色對於數據的訪問許可權都應該大相徑庭。 列如 1、不同登錄人員對一個數據列表的可見度是不一樣的,如數據列、數據行、數據按鈕 ...
  • 前言 上一篇文章寫瞭如何使用RabbitMQ做個簡單的發送郵件項目,然後評論也是比較多,也是準備去學習一下如何確保RabbitMQ的消息可靠性,但是由於時間原因,先來說說設計模式中的簡單工廠模式吧! 在瞭解簡單工廠模式之前,我們要知道C#是一款面向對象的高級程式語言。它有3大特性,封裝、繼承、多態。 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 介紹 Nodify是一個WPF基於節點的編輯器控制項,其中包含一系列節點、連接和連接器組件,旨在簡化構建基於節點的工具的過程 ...
  • 創建一個webapi項目做測試使用。 創建新控制器,搭建一個基礎框架,包括獲取當天日期、wiki的請求地址等 創建一個Http請求幫助類以及方法,用於獲取指定URL的信息 使用http請求訪問指定url,先運行一下,看看返回的內容。內容如圖右邊所示,實際上是一個Json數據。我們主要解析 大事記 部 ...
  • 最近在不少自媒體上看到有關.NET與C#的資訊與評價,感覺大家對.NET與C#還是不太瞭解,尤其是對2016年6月發佈的跨平臺.NET Core 1.0,更是知之甚少。在考慮一番之後,還是決定寫點東西總結一下,也回顧一下.NET的發展歷史。 首先,你沒看錯,.NET是跨平臺的,可以在Windows、 ...
  • Nodify學習 一:介紹與使用 - 可樂_加冰 - 博客園 (cnblogs.com) Nodify學習 二:添加節點 - 可樂_加冰 - 博客園 (cnblogs.com) 添加節點(nodes) 通過上一篇我們已經創建好了編輯器實例現在我們為編輯器添加一個節點 添加model和viewmode ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...
  • 類型檢查和轉換:當你需要檢查對象是否為特定類型,並且希望在同一時間內將其轉換為那個類型時,模式匹配提供了一種更簡潔的方式來完成這一任務,避免了使用傳統的as和is操作符後還需要進行額外的null檢查。 複雜條件邏輯:在處理複雜的條件邏輯時,特別是涉及到多個條件和類型的情況下,使用模式匹配可以使代碼更 ...
  • 在日常開發中,我們經常需要和文件打交道,特別是桌面開發,有時候就會需要載入大批量的文件,而且可能還會存在部分文件缺失的情況,那麼如何才能快速的判斷文件是否存在呢?如果處理不當的,且文件數量比較多的時候,可能會造成卡頓等情況,進而影響程式的使用體驗。今天就以一個簡單的小例子,簡述兩種不同的判斷文件是否... ...
  • 前言 資料庫併發,數據審計和軟刪除一直是數據持久化方面的經典問題。早些時候,這些工作需要手寫複雜的SQL或者通過存儲過程和觸發器實現。手寫複雜SQL對軟體可維護性構成了相當大的挑戰,隨著SQL字數的變多,用到的嵌套和複雜語法增加,可讀性和可維護性的難度是幾何級暴漲。因此如何在實現功能的同時控制這些S ...