0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

算法時(shí)空復(fù)雜度分析實(shí)用指南(下)

jf_78858299 ? 來(lái)源:labuladong ? 作者:labuladong ? 2023-04-19 10:35 ? 次閱讀

遞歸算法分析

對(duì)很多人來(lái)說(shuō),遞歸算法的時(shí)間復(fù)雜度是比較難分析的。但如果你有 框架思維,明白所有遞歸算法的本質(zhì)是樹(shù)的遍歷,那么分析起來(lái)應(yīng)該沒(méi)什么難度。

計(jì)算算法的時(shí)間復(fù)雜度,無(wú)非就是看這個(gè)算法做了些啥事兒,花了多少時(shí)間。而遞歸算法做的事情就是遍歷一棵遞歸樹(shù),在樹(shù)上的每個(gè)節(jié)點(diǎn)所做一些事情罷了。

所以:

遞歸算法的時(shí)間復(fù)雜度 = 遞歸的次數(shù) x 函數(shù)本身的時(shí)間復(fù)雜度

遞歸算法的空間復(fù)雜度 = 遞歸堆棧的深度 + 算法申請(qǐng)的存儲(chǔ)空間

或者再說(shuō)得直觀一點(diǎn):

遞歸算法的時(shí)間復(fù)雜度 = 遞歸樹(shù)的節(jié)點(diǎn)個(gè)數(shù) x 每個(gè)節(jié)點(diǎn)的時(shí)間復(fù)雜度

遞歸算法的空間復(fù)雜度 = 遞歸樹(shù)的高度 + 算法申請(qǐng)的存儲(chǔ)空間

函數(shù)遞歸的原理是操作系統(tǒng)維護(hù)的函數(shù)堆棧,所以遞歸棧的空間消耗也需要算在空間復(fù)雜度之內(nèi),這一點(diǎn)不要忘了。

首先說(shuō)一下動(dòng)態(tài)規(guī)劃算法 ,還是拿前文 動(dòng)態(tài)規(guī)劃核心框架 中講到的湊零錢問(wèn)題舉例,它的暴力遞歸解法主體如下:

int dp(int[] coins, int amount) {
    // base case
    if (amount == 0) return 0;
    if (amount < 0) return -1;

    int res = Integer.MAX_VALUE;
    // 時(shí)間 O(K)
    for (int coin : coins) {
        int subProblem = dp(coins, amount - coin);
        if (subProblem == -1) continue;
        res = Math.min(res, subProblem + 1);
    }

    return res == Integer.MAX_VALUE ? -1 : res;
}

當(dāng)amount = 11, coins = [1,2,5]時(shí),該算法的遞歸樹(shù)就長(zhǎng)這樣:

圖片

剛才說(shuō)了這棵樹(shù)上的節(jié)點(diǎn)個(gè)數(shù)為O(K^N),那么每個(gè)節(jié)點(diǎn)消耗的時(shí)間復(fù)雜度是多少呢?其實(shí)就是這個(gè)dp函數(shù)本身的復(fù)雜度。

你看dp函數(shù)里面有個(gè) for 循環(huán)遍歷長(zhǎng)度為Kcoins列表,所以函數(shù)本身的復(fù)雜度為O(K),故該算法總的時(shí)間復(fù)雜度為:

O(K^N) * O(K) = O(K^(N+1))

當(dāng)然,之前也說(shuō)了,這個(gè)復(fù)雜度只是一個(gè)粗略的上界,并不準(zhǔn)確,真實(shí)的效率肯定會(huì)高一些。

這個(gè)算法的空間復(fù)雜度很容易分析:

dp函數(shù)本身沒(méi)有申請(qǐng)數(shù)組之類的,所以算法申請(qǐng)的存儲(chǔ)空間為O(1);而dp函數(shù)的堆棧深度為遞歸樹(shù)的高度O(N),所以這個(gè)算法的空間復(fù)雜度為O(N)

暴力遞歸解法的分析結(jié)束,但這個(gè)解法存在重疊子問(wèn)題,通過(guò)備忘錄消除重疊子問(wèn)題的冗余計(jì)算之后,相當(dāng)于在原來(lái)的遞歸樹(shù)上進(jìn)行剪枝:

// 備忘錄,空間 O(N)
memo = new int[N];
Arrays.fill(memo, -666);

int dp(int[] coins, int amount) {
    if (amount == 0) return 0;
    if (amount < 0) return -1;
    // 查備忘錄,防止重復(fù)計(jì)算
    if (memo[amount] != -666)
        return memo[amount];

    int res = Integer.MAX_VALUE;
    // 時(shí)間 O(K)
    for (int coin : coins) {
        int subProblem = dp(coins, amount - coin);
        if (subProblem == -1) continue;
        res = Math.min(res, subProblem + 1);
    }
    // 把計(jì)算結(jié)果存入備忘錄
    memo[amount] = (res == Integer.MAX_VALUE) ? -1 : res;
    return memo[amount];
}

通過(guò)備忘錄剪掉了大量節(jié)點(diǎn)之后,雖然函數(shù)本身的時(shí)間復(fù)雜度依然是O(K),但大部分遞歸在函數(shù)開(kāi)頭就立即返回了,根本不會(huì)執(zhí)行到 for 循環(huán)那里,所以可以認(rèn)為遞歸函數(shù)執(zhí)行的次數(shù)(遞歸樹(shù)上的節(jié)點(diǎn))減少了,從而時(shí)間復(fù)雜度下降。

剪枝之后還剩多少節(jié)點(diǎn)呢?根據(jù)備忘錄剪枝的原理,相同「狀態(tài)」不會(huì)被重復(fù)計(jì)算,所以剪枝之后剩下的節(jié)點(diǎn)數(shù)就是「狀態(tài)」的數(shù)量,即memo的大小N

所以,對(duì)于帶備忘錄的動(dòng)態(tài)規(guī)劃算法的時(shí)間復(fù)雜度,以下幾種理解方式都是等價(jià)的:

遞歸的次數(shù) x 函數(shù)本身的時(shí)間復(fù)雜度
= 遞歸樹(shù)節(jié)點(diǎn)個(gè)數(shù) x 每個(gè)節(jié)點(diǎn)的時(shí)間復(fù)雜度
= 狀態(tài)個(gè)數(shù) x 計(jì)算每個(gè)狀態(tài)的時(shí)間復(fù)雜度
= 子問(wèn)題個(gè)數(shù) x 解決每個(gè)子問(wèn)題的時(shí)間復(fù)雜度
= O(N) * O(K)
= O(NK)

像「狀態(tài)」「子問(wèn)題」屬于動(dòng)態(tài)規(guī)劃類型問(wèn)題特有的詞匯,但時(shí)間復(fù)雜度本質(zhì)上還是遞歸次數(shù) x 函數(shù)本身復(fù)雜度,換湯不換藥罷了。反正你愛(ài)怎么說(shuō)怎么說(shuō)吧,別把自己繞進(jìn)去就行。

備忘錄優(yōu)化解法的空間復(fù)雜度也不難分析:

dp函數(shù)的堆棧深度為「狀態(tài)」的個(gè)數(shù),依然是O(N),而算法申請(qǐng)了一個(gè)大小為O(N)的備忘錄memo數(shù)組,所以總的空間復(fù)雜度為O(N) + O(N) = O(N)。

雖然用 Big O 表示法來(lái)看,優(yōu)化前后的空間復(fù)雜度相同,不過(guò)顯然優(yōu)化解法消耗的空間要更多,所以用備忘錄進(jìn)行剪枝也被稱為「用空間換時(shí)間」。

如果你把自頂向下帶備忘錄的解法進(jìn)一步改寫成自底向上的迭代解法:

int coinChange(int[] coins, int amount) {
    // 空間 O(N)
    int[] dp = new int[amount + 1];
    Arrays.fill(dp, amount + 1);

    dp[0] = 0;
    // 時(shí)間 O(KN)
    for (int i = 0; i < dp.length; i++) {
        for (int coin : coins) {
            if (i - coin < 0) continue;
            dp[i] = Math.min(dp[i], 1 + dp[i - coin]);
        }
    }
    return (dp[amount] == amount + 1) ? -1 : dp[amount];
}

該解法的時(shí)間復(fù)雜度不變,但已經(jīng)不存在遞歸,所以空間復(fù)雜度中不需要考慮堆棧的深度,只需考慮dp數(shù)組的存儲(chǔ)空間,雖然用 Big O 表示法來(lái)看,該算法的空間復(fù)雜度依然是O(N),但該算法的實(shí)際空間消耗是更小的,所以自底向上迭代的動(dòng)態(tài)規(guī)劃是各方面性能最好的。

接下來(lái)說(shuō)一下回溯算法 ,需要你看過(guò)前文回溯算法秒殺排列組合問(wèn)題的 9 種變體,下面我會(huì)以標(biāo)準(zhǔn)的全排列問(wèn)題和子集問(wèn)題的解法為例,分析一下其時(shí)間復(fù)雜度。

先看標(biāo)準(zhǔn)全排列問(wèn)題 (元素?zé)o重不可復(fù)選)的核心函數(shù)backtrack

// 回溯算法計(jì)算全排列
void backtrack(int[] nums) {
    // 到達(dá)葉子節(jié)點(diǎn),收集路徑值,時(shí)間 O(N)
    if (track.size() == nums.length) {
        res.add(new LinkedList(track));
        return;
    }

    // 非葉子節(jié)點(diǎn),遍歷所有子節(jié)點(diǎn),時(shí)間 O(N)
    for (int i = 0; i < nums.length; i++) {
        if (used[i]) {
            // 剪枝邏輯
            continue;
        }
        // 做選擇
        used[i] = true;
        track.addLast(nums[i]);
        backtrack(nums);
        // 取消選擇
        track.removeLast();
        used[i] = false;
    }
}

當(dāng)nums = [1,2,3]時(shí),backtrack其實(shí)在遍歷這棵遞歸樹(shù):

圖片

假設(shè)輸入的nums數(shù)組長(zhǎng)度為N,那么這個(gè)backtrack函數(shù)遞歸了多少次?backtrack函數(shù)本身的復(fù)雜度是多少?

先看看backtrack函數(shù)本身的時(shí)間復(fù)雜度,即樹(shù)中每個(gè)節(jié)點(diǎn)的復(fù)雜度。

對(duì)于非葉子節(jié)點(diǎn),會(huì)執(zhí)行 for 循環(huán),復(fù)雜度為O(N);對(duì)于葉子節(jié)點(diǎn),不會(huì)執(zhí)行循環(huán),但將track中的值拷貝到res列表中也需要O(N)的時(shí)間, 所以backtrack函數(shù)本身的時(shí)間復(fù)雜度為O(N) 。

PS:函數(shù)本身(每個(gè)節(jié)點(diǎn))的時(shí)間復(fù)雜度并不是樹(shù)枝的條數(shù)??创a,每個(gè)節(jié)點(diǎn)都會(huì)執(zhí)行整個(gè) for 循環(huán),所以每個(gè)節(jié)點(diǎn)的復(fù)雜度都是O(N)

再來(lái)看看backtrack函數(shù)遞歸了多少次,即這個(gè)排列樹(shù)上有多少個(gè)節(jié)點(diǎn)。

第 0 層(根節(jié)點(diǎn))有P(N, 0) = 1個(gè)節(jié)點(diǎn)。

第 1 層有P(N, 1) = N個(gè)節(jié)點(diǎn)。

第 2 層有P(N, 2) = N x (N - 1)個(gè)節(jié)點(diǎn)。

第 3 層有P(N, 3) = N x (N - 1) x (N - 2)個(gè)節(jié)點(diǎn)。

以此類推,其中P就是我們高中學(xué)過(guò)的排列數(shù)函數(shù)。

全排列的回溯樹(shù)高度為N,所以節(jié)點(diǎn)總數(shù)為:

P(N, 0) + P(N, 1) + P(N, 2) + ... + P(N, N)

這一堆排列數(shù)累加不好算,粗略估計(jì)一下上界吧,把它們?nèi)紨U(kuò)大成P(N, N) = N!那么節(jié)點(diǎn)總數(shù)的上界就是O(N*N!) 。

現(xiàn)在就可以得出算法的總時(shí)間復(fù)雜度:

遞歸的次數(shù) x 函數(shù)本身的時(shí)間復(fù)雜度
= 遞歸樹(shù)節(jié)點(diǎn)個(gè)數(shù) x 每個(gè)節(jié)點(diǎn)的時(shí)間復(fù)雜度
= O(N*N!) * O(N)
= O(N^2 * N!)

當(dāng)然,由于計(jì)算節(jié)點(diǎn)總數(shù)的時(shí)候我們?yōu)榱朔奖阌?jì)算把累加項(xiàng)擴(kuò)大了很多,所以這個(gè)結(jié)果肯定也是偏大的,不過(guò)用來(lái)描述復(fù)雜度的上界還是可以接受的。

分析下該算法的空間復(fù)雜度:

backtrack函數(shù)的遞歸深度為遞歸樹(shù)的高度O(N),而算法需要存儲(chǔ)所有全排列的結(jié)果,即需要申請(qǐng)的空間為O(N*N!)所以總的空間復(fù)雜度為O(N*N!) 。

最后看下標(biāo)準(zhǔn)子集問(wèn)題 (元素?zé)o重不可復(fù)選)的核心函數(shù)backtrack

// 回溯算法計(jì)算所有子集(冪集)
void backtrack(int[] nums, int start) {

    // 每個(gè)節(jié)點(diǎn)的值都是一個(gè)子集,O(N)
    res.add(new LinkedList<>(track));

    // 遍歷子節(jié)點(diǎn),O(N)
    for (int i = start; i < nums.length; i++) {
        // 做選擇
        track.addLast(nums[i]);
        backtrack(nums, i + 1);
        // 撤銷選擇
        track.removeLast();
    }
}

當(dāng)nums = [1,2,3]時(shí),backtrack其實(shí)在遍歷這棵遞歸樹(shù):

假設(shè)輸入的nums數(shù)組長(zhǎng)度為N,那么這個(gè)backtrack函數(shù)遞歸了多少次?backtrack函數(shù)本身的復(fù)雜度是多少?

先看看backtrack函數(shù)本身的時(shí)間復(fù)雜度,即樹(shù)中每個(gè)節(jié)點(diǎn)的復(fù)雜度。

backtrack函數(shù)在前序位置都會(huì)將track列表拷貝到res中,消耗O(N)的時(shí)間,且會(huì)執(zhí)行一個(gè) for 循環(huán),也消耗O(N)的時(shí)間, 所以backtrack函數(shù)本身的時(shí)間復(fù)雜度為O(N) 。

再來(lái)看看backtrack函數(shù)遞歸了多少次,即這個(gè)排列樹(shù)上有多少個(gè)節(jié)點(diǎn)。

那就直接看圖一層一層數(shù)唄:

第 0 層(根節(jié)點(diǎn))有C(N, 0) = 1個(gè)節(jié)點(diǎn)。

第 1 層有C(N, 1) = N個(gè)節(jié)點(diǎn)。

第 2 層有C(N, 2)個(gè)節(jié)點(diǎn)。

第 3 層有C(N, 3)個(gè)節(jié)點(diǎn)。

以此類推,其中C就是我們高中學(xué)過(guò)的組合數(shù)函數(shù)。

由于這棵組合樹(shù)的高度為N,組合數(shù)求和公式是高中學(xué)過(guò)的, 所以總的節(jié)點(diǎn)數(shù)為2^N

C(N, 0) + C(N, 1) + C(N, 2) + ... + C(N, N) = 2^N

就算你忘記了組合數(shù)求和公式,其實(shí)也可以推導(dǎo)出來(lái)節(jié)點(diǎn)總數(shù):因?yàn)?code>N個(gè)元素的所有子集(冪集)數(shù)量為2^N,而這棵樹(shù)的每個(gè)節(jié)點(diǎn)代表一個(gè)子集,所以樹(shù)的節(jié)點(diǎn)總數(shù)也為2^N。

那么,現(xiàn)在就可以得出算法的總復(fù)雜度:

遞歸的次數(shù) x 函數(shù)本身的時(shí)間復(fù)雜度
= 遞歸樹(shù)節(jié)點(diǎn)個(gè)數(shù) x 每個(gè)節(jié)點(diǎn)的時(shí)間復(fù)雜度
= O(2^N) * O(N)
= O(N*2^N)

分析下該算法的空間復(fù)雜度:

backtrack函數(shù)的遞歸深度為遞歸樹(shù)的高度O(N),而算法需要存儲(chǔ)所有子集的結(jié)果,粗略估算下需要申請(qǐng)的空間為O(N*2^N), 所以總的空間復(fù)雜度為O(N*2^N) 。

到這里,標(biāo)準(zhǔn)排列/子集問(wèn)題的時(shí)間復(fù)雜度就分析完了,前文 回溯算法秒殺排列組合問(wèn)題的 9 種變體中的其他問(wèn)題變形都可以按照類似的邏輯分析,這些就留給你自己分析吧。

最后總結(jié)

本文篇幅較大,我簡(jiǎn)單總結(jié)下重點(diǎn):

1、Big O 標(biāo)記代表一個(gè)函數(shù)的集合,用它表示時(shí)空復(fù)雜度時(shí)代表一個(gè)上界,所以如果你和別人算的復(fù)雜度不一樣,可能你們都是對(duì)的,只是精確度不同罷了。

2、時(shí)間復(fù)雜度的分析不難,關(guān)鍵是你要透徹理解算法到底干了什么事。非遞歸算法中嵌套循環(huán)的復(fù)雜度依然可能是線性的;數(shù)據(jù)結(jié)構(gòu) API 需要用平均時(shí)間復(fù)雜度衡量性能;遞歸算法本質(zhì)是遍歷遞歸樹(shù),時(shí)間復(fù)雜度取決于遞歸樹(shù)中節(jié)點(diǎn)的個(gè)數(shù)(遞歸次數(shù))和每個(gè)節(jié)點(diǎn)的復(fù)雜度(遞歸函數(shù)本身的復(fù)雜度)。

好了,能看到這里,真得給你鼓掌。需要說(shuō)明的是,本文給出的一些復(fù)雜度都是比較粗略的估算,上界都不是很「緊」,如果你不滿足于粗略的估算,想計(jì)算更「緊」更精確的上界,就需要比較好的數(shù)學(xué)功底了。不過(guò)從面試筆試的角度來(lái)說(shuō),掌握這些基本分析技術(shù)已經(jīng)足夠應(yīng)對(duì)了。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    基于紋理復(fù)雜度的快速幀內(nèi)預(yù)測(cè)算法

    【正文快照】:0引言幀內(nèi)編碼利用相鄰像素塊之間的相關(guān)[1]來(lái)減少視頻圖像的空間冗余,提高了編碼效率。但是在H.264/AVC的幀內(nèi)預(yù)測(cè)采用全搜索算法中,為了確定一個(gè)宏塊的最優(yōu)預(yù)測(cè)模式,要遍歷色度塊和亮度塊的17種預(yù)測(cè)模式,計(jì)算率失真代價(jià)值的并比較大小,是造成H.264運(yùn)
    發(fā)表于 05-06 09:01

    時(shí)間復(fù)雜度是指什么

    原理->微機(jī)原理->軟件工程,編譯原理,數(shù)據(jù)庫(kù)數(shù)據(jù)結(jié)構(gòu)1.時(shí)間復(fù)雜度時(shí)間復(fù)雜度是指執(zhí)行算法所需要的計(jì)算工作量,因?yàn)檎麄€(gè)算法的執(zhí)行時(shí)間與基本操作重復(fù)執(zhí)行的...
    發(fā)表于 07-22 10:01

    各種排序算法的時(shí)間空間復(fù)雜度、穩(wěn)定性

    各種排序算法的時(shí)間空間復(fù)雜度、穩(wěn)定性一、排序算法分類:二、排序算法比較:注:1、歸并排序可以通過(guò)手搖算法將空間
    發(fā)表于 12-21 07:48

    LDPC碼低復(fù)雜度譯碼算法研究

    在描述置信傳播(BP)譯碼算法基礎(chǔ)上, 研究和分析了兩種降低復(fù)雜度的譯碼算法。Min.Sum 算法主要討論了簡(jiǎn)化校驗(yàn)節(jié)點(diǎn)的消息更新運(yùn)算,并應(yīng)
    發(fā)表于 03-31 15:22 ?7次下載
    LDPC碼低<b class='flag-5'>復(fù)雜度</b>譯碼<b class='flag-5'>算法</b>研究

    基于復(fù)雜度分析的改進(jìn)A_算法飛行器航跡規(guī)劃_叢林虎

    基于復(fù)雜度分析的改進(jìn)A_算法飛行器航跡規(guī)劃_叢林虎
    發(fā)表于 03-17 15:11 ?0次下載

    圖像復(fù)雜度對(duì)信息隱藏性能影響分析

    算法進(jìn)行實(shí)驗(yàn),研究圖像的復(fù)雜度差異對(duì)信息隱藏性能的影響。實(shí)驗(yàn)結(jié)果表明了所提復(fù)雜度評(píng)價(jià)方法的有效性以及復(fù)雜度分類的合理性,依據(jù)圖像復(fù)雜度準(zhǔn)則對(duì)
    發(fā)表于 11-14 09:57 ?5次下載

    虛擬MIMO中低復(fù)雜度功率分配算法

    一種基于線性注水原理的低復(fù)雜度功率分配算法。該算法通過(guò)快速排除信道條件較差的協(xié)作用戶,并利用各協(xié)作用戶功率值之間的線性遞推關(guān)系式,將最優(yōu)功率分配算法中的迭代運(yùn)算轉(zhuǎn)化為線性運(yùn)算,在實(shí)現(xiàn)功
    發(fā)表于 03-09 15:22 ?1次下載
    虛擬MIMO中低<b class='flag-5'>復(fù)雜度</b>功率分配<b class='flag-5'>算法</b>

    如何求遞歸算法的時(shí)間復(fù)雜度

    那么我通過(guò)一道簡(jiǎn)單的面試題,模擬面試的場(chǎng)景,來(lái)帶大家逐步分析遞歸算法的時(shí)間復(fù)雜度,最后找出最優(yōu)解,來(lái)看看同樣是遞歸,怎么就寫成了O(n)的代碼。
    的頭像 發(fā)表于 07-13 11:30 ?2289次閱讀

    如何求遞歸算法的時(shí)間復(fù)雜度

    相信很多同學(xué)對(duì)遞歸算法的時(shí)間復(fù)雜度都很模糊,那么這篇Carl來(lái)給大家通透的講一講。
    的頭像 發(fā)表于 07-13 11:33 ?1637次閱讀

    算法之空間復(fù)雜度

    算法之空間復(fù)雜度:衡量一個(gè)算法運(yùn)行需要開(kāi)辟的額外空間
    的頭像 發(fā)表于 08-31 10:29 ?1633次閱讀

    常見(jiàn)機(jī)器學(xué)習(xí)算法的計(jì)算復(fù)雜度

    時(shí)間復(fù)雜度不是測(cè)量一個(gè)算法或一段代碼在某個(gè)機(jī)器或者條件運(yùn)行所花費(fèi)的時(shí)間。時(shí)間復(fù)雜度一般指時(shí)間復(fù)雜性,時(shí)間
    發(fā)表于 10-02 12:45 ?822次閱讀

    算法時(shí)空復(fù)雜度分析實(shí)用指南1

    我以前的文章主要都是講解算法的原理和解題的思維,對(duì)時(shí)間復(fù)雜度和空間復(fù)雜度分析經(jīng)常一筆帶過(guò),主要是基于以下兩個(gè)原因:
    的頭像 發(fā)表于 04-12 14:37 ?535次閱讀
    <b class='flag-5'>算法</b><b class='flag-5'>時(shí)空</b><b class='flag-5'>復(fù)雜度</b><b class='flag-5'>分析</b>實(shí)用<b class='flag-5'>指南</b>1

    算法時(shí)空復(fù)雜度分析實(shí)用指南2

    類似的,想想之前說(shuō)的數(shù)據(jù)結(jié)構(gòu)擴(kuò)容的場(chǎng)景,也許`N`次操作中的某一次操作恰好觸發(fā)了擴(kuò)容,導(dǎo)致時(shí)間復(fù)雜度提高,但總的時(shí)間復(fù)雜度依然保持在`O(N)`,所以均攤到每一次操作上,其平均時(shí)間復(fù)雜度依然是`O(1)`。
    的頭像 發(fā)表于 04-12 14:38 ?553次閱讀
    <b class='flag-5'>算法</b><b class='flag-5'>時(shí)空</b><b class='flag-5'>復(fù)雜度</b><b class='flag-5'>分析</b>實(shí)用<b class='flag-5'>指南</b>2

    算法時(shí)空復(fù)雜度分析實(shí)用指南(上)

    本文會(huì)篇幅較長(zhǎng),會(huì)涵蓋如下幾點(diǎn): 1、Big O 表示法的幾個(gè)基本特點(diǎn)。 2、非遞歸算法中的時(shí)間復(fù)雜度分析。 3、數(shù)據(jù)結(jié)構(gòu) API 的效率衡量方法(攤還分析)。
    的頭像 發(fā)表于 04-19 10:34 ?870次閱讀
    <b class='flag-5'>算法</b><b class='flag-5'>時(shí)空</b><b class='flag-5'>復(fù)雜度</b><b class='flag-5'>分析</b>實(shí)用<b class='flag-5'>指南</b>(上)

    如何計(jì)算時(shí)間復(fù)雜度

    1 算法與時(shí)間復(fù)雜度 算法(Algorithm)是求解一個(gè)問(wèn)題需要遵循的,被清楚指定的簡(jiǎn)單指令的集合。 算法一旦確定,那么下一步就要確定該算法
    的頭像 發(fā)表于 10-13 11:19 ?3045次閱讀
    如何計(jì)算時(shí)間<b class='flag-5'>復(fù)雜度</b>