關(guān)于時間復(fù)雜度介紹


知道算法的速度和它需要多少空間是很有用的血公。這允許您為工作選擇正確的算法。
O符號給出了一個算法運(yùn)行時間和它使用的內(nèi)存量的粗略指示馏段。當(dāng)有人說, "這個算法有最壞的情況下運(yùn)行時間 O(n^2)和使用 O(n) 空間," 他們的意思是它有點(diǎn)慢, 但不需要大量額外的內(nèi)存堰塌。
計算算法的O通常是通過數(shù)學(xué)分析來完成的涡拘。我們在這里跳數(shù)學(xué),但是, 知道不同的值意味著什么是有用的, 所以這里有一個方便的表疼电。n是指正在處理的數(shù)據(jù)項的數(shù)量嚼锄。例如,當(dāng)排序100個項目的數(shù)組時蔽豺,n=100区丑。

下面是每一類性能的一些例子:

Big-O Name Description
O(1) 恒久不變(constant) 這是最好的算法 不管有多少數(shù)據(jù)算法總是需要相同的時間量。示例:用索引查找數(shù)組的元素修陡。
O(log n) 對數(shù)(logarithmic) 非常好 這些算法在每次迭代中將數(shù)據(jù)量減半沧侥。如果你有100個項目,需要大約7個步驟才能找到答案魄鸦。有1000項, 它需要10個步驟宴杀,1000000項只采取20個步驟。即使對于大量的數(shù)據(jù)拾因,這也是非惩眨快的。示例:二進(jìn)制搜索绢记。
O(n) 線性(linear) 良好性能 如果你有100項, 這是100單位的工作扁达,將項目的數(shù)量加倍使得算法花費(fèi)了兩倍長(200個工作單位)。例子:順序搜索蠢熄。
O(n log n) 線性化(linearithmic) 體面的表現(xiàn) 這比線性略差跪解,但也不壞。例子:最快的通用排序算法护赊。
O(n^2) 平方(quadratic) 有點(diǎn)慢 如果你有100個項目惠遏,這就等于100 ^ 2=10000個單位的工作。將項目數(shù)加倍使其慢四倍(因為2的平方等于4)骏啰。示例:使用嵌套循環(huán)的算法节吮,如插入排序、冒泡排序判耕。
O(n^3) 立方(cubic) 性能不佳 如果你有100項, 這做 100 ^ 3 = 100萬單位的工作透绩。加倍輸入大小使其慢了八倍。例子:矩陣乘法壁熄。
O(n^3) 立方(cubic) 性能不佳 如果你有100項, 這做 100 ^ 3 = 100萬單位的工作帚豪。加倍輸入大小使其慢了八倍。例子:矩陣乘法草丧。
O(2^n)) 指數(shù)級(exponential) 性能非常差 你想避免這些算法狸臣,但有時你別無選擇。只需在輸入中添加一位即可使運(yùn)行時間加倍昌执。示例: 旅行推銷員問題烛亦。
O(n!) 階乘(factorial) 難以忍受的緩慢 做任何事情都需要一百萬年的時間诈泼。

O(1)

O(1)復(fù)雜性最常見的例子是訪問數(shù)組索引。

let value = array[5]

O(1)的另一個例子是從棧中推送和彈出煤禽。

O(log n)

var j = 1
while j < n {
  // do constant time stuff
  j *= 2
}

"j" 不是簡單地遞增, 而是在每次運(yùn)行時增加2倍铐达。
二進(jìn)制搜索算法是O(log n)復(fù)雜性的一個例子。

O(n)

for i in stride(from: 0, to: n, by: 1) {
  print(array[I])
}

數(shù)組遍歷和線性搜索是O(n)復(fù)雜度的例子檬果。

O(n log n)

for i in stride(from: 0, to: n, by: 1) {
var j = 1
  while j < n {
    j *= 2
    // do constant time stuff
  }
}

或者

for i in stride(from: 0, to: n, by: 1) {
  func index(after i: Int) -> Int? { // multiplies `i` by 2 until `i` >= `n`
    return i < n ? i * 2 : nil
  }
  for j in sequence(first: 1, next: index(after:)) {
    // do constant time stuff
  }
}

合并排序和堆排序是O(n log n)復(fù)雜性的示例瓮孙。

O(n^2)

for i  in stride(from: 0, to: n, by: 1) {
  for j in stride(from: 1, to: n, by: 1) {
    // do constant time stuff
  }
}

遍歷一個簡單的二維數(shù)組和冒泡排序是O(n^2)復(fù)雜度的例子。

O(n^3)

for i in stride(from: 0, to: n, by: 1) {
  for j in stride(from: 1, to: n, by: 1) {
    for k in stride(from: 1, to: n, by: 1) {
      // do constant time stuff
    }
  }
}

O(2^n)

運(yùn)行時間O(2^N)的算法通常是遞歸算法选脊,通過遞歸求解兩個較小的n-1個問題來解決n的問題杭抠。下面的示例打印出解決 N 個磁盤上著名的 "河內(nèi)塔" 問題所需的所有動作。

func solveHanoi(n: Int, from: String, to: String, spare: String) {
  guard n >= 1 else { return }
  if n > 1 {
    solveHanoi(n: n - 1, from: from, to: spare, spare: to)
  } else {
    solveHanoi(n: n - 1, from: spare, to: to, spare: from)
  }
}

O(n!)

函數(shù)的最平凡例子知牌,取O(n!)時間如下祈争。

func nFactFunc(n: Int) {
  for i in stride(from: 0, to: n, by: 1) {
    nFactFunc(n: n - 1)
  }
}

通常, 你不需要數(shù)學(xué)來計算時間復(fù)雜度是多少, 但你可以簡單地使用你的直覺。如果你的代碼使用一個單循環(huán)來查看輸入的所有n個元素角寸,則算法是O(n)菩混。果代碼有兩個嵌套循環(huán),則為O(n^2)扁藕。三個嵌套循環(huán)給出O(n^3)沮峡,以此類推。
請注意, O 表示法是一個估計, 只是真正有用的大值 n亿柑。例如邢疙,插入排序算法的最壞情況運(yùn)行時間是O(n^2)。在理論上, 比合并排序的運(yùn)行時間差, 即O(n log n)望薄。但是對于少量數(shù)據(jù)疟游,插入排序?qū)嶋H上更快,特別是如果數(shù)組已經(jīng)部分排序了痕支。
如果你覺得這很混亂颁虐,不要讓這個O打擾你太多。當(dāng)比較兩種算法來找出哪一種算法更好時卧须,這是最有用的另绩。但最終你還是想在實踐中檢驗?zāi)囊粋€才是最好的。如果數(shù)據(jù)量相對較小花嘶,那么即使是慢算法也將足夠快用于實際使用笋籽。

傳送門

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市椭员,隨后出現(xiàn)的幾起案子车海,更是在濱河造成了極大的恐慌,老刑警劉巖隘击,帶你破解...
    沈念sama閱讀 219,539評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件侍芝,死亡現(xiàn)場離奇詭異喘沿,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)竭贩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評論 3 396
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來莺禁,“玉大人留量,你說我怎么就攤上這事∮炊” “怎么了楼熄?”我有些...
    開封第一講書人閱讀 165,871評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長浩峡。 經(jīng)常有香客問我可岂,道長,這世上最難降的妖魔是什么翰灾? 我笑而不...
    開封第一講書人閱讀 58,963評論 1 295
  • 正文 為了忘掉前任缕粹,我火速辦了婚禮,結(jié)果婚禮上纸淮,老公的妹妹穿的比我還像新娘平斩。我一直安慰自己,他們只是感情好咽块,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,984評論 6 393
  • 文/花漫 我一把揭開白布绘面。 她就那樣靜靜地躺著,像睡著了一般侈沪。 火紅的嫁衣襯著肌膚如雪揭璃。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,763評論 1 307
  • 那天亭罪,我揣著相機(jī)與錄音瘦馍,去河邊找鬼。 笑死皆撩,一個胖子當(dāng)著我的面吹牛扣墩,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播扛吞,決...
    沈念sama閱讀 40,468評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼呻惕,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了滥比?” 一聲冷哼從身側(cè)響起亚脆,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎盲泛,沒想到半個月后濒持,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體键耕,經(jīng)...
    沈念sama閱讀 45,850評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,002評論 3 338
  • 正文 我和宋清朗相戀三年柑营,在試婚紗的時候發(fā)現(xiàn)自己被綠了屈雄。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,144評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡官套,死狀恐怖酒奶,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情奶赔,我是刑警寧澤惋嚎,帶...
    沈念sama閱讀 35,823評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站站刑,受9級特大地震影響另伍,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜绞旅,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,483評論 3 331
  • 文/蒙蒙 一摆尝、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧玻靡,春花似錦结榄、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至蝎土,卻和暖如春视哑,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背誊涯。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評論 1 272
  • 我被黑心中介騙來泰國打工挡毅, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人暴构。 一個月前我還...
    沈念sama閱讀 48,415評論 3 373
  • 正文 我出身青樓跪呈,卻偏偏與公主長得像,于是被迫代替她去往敵國和親取逾。 傳聞我的和親對象是個殘疾皇子耗绿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,092評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 第2章 基本語法 2.1 概述 基本句法和變量 語句 JavaScript程序的執(zhí)行單位為行(line),也就是一...
    悟名先生閱讀 4,150評論 0 13
  • --- layout: post title: "如果有人問你關(guān)系型數(shù)據(jù)庫的原理砾隅,叫他看這篇文章(轉(zhuǎn))" date...
    藍(lán)墜星閱讀 793評論 0 3
  • 世俗總強(qiáng)迫我 說還在乎你 其實我們都知道 我不愛別人 當(dāng)然误阻,也不愛你 我不是媚俗的芍藥 自然不能許你今世來生 我是...
    咚塔塔族族人閱讀 224評論 0 2
  • 生活就像老女傭,叫她找一樣?xùn)|西,她總要慢條斯理的從大抽屜里取出一個花格子小手巾包究反,去掉別針寻定,打開來輕輕掀著看了一遍...
    川貝貝閱讀 683評論 2 5
  • 第三章.異能者(3) "不要躲了,你以為你逃得了么精耐?...
    千盈落殤閱讀 129評論 0 0