- F2 l& e" `5 H2 V; R4)多个规模求加法:比如方法有两个参数控制两个循环的次数,那么这时就取二者复杂度相加。0 p' f- X3 _5 K
0 y& l7 c% P; p" m# T
四、常用的复杂度级别? 3 c" |# `) T5 a多项式阶:随着数据规模的增长,算法的执行时间和空间占用,按照多项式的比例增长。包括, O(1)(常数阶)、O(logn)(对数阶)、O(n)(线性阶)、O(nlogn)(线性对数阶)、O(n^2 ) (平方阶)、O( n^3)(立方阶)! O! n, h8 Z/ ]
w( c/ k8 Z0 G! Q$ Z
非多项式阶:随着数据规模的增长,算法的执行时间和空间占用暴增,这类算法性能极差。包括, O( 2^n )(指数阶)、O(n!)(阶乘阶)" r3 U- v: `1 o3 s S
. Q7 j8 _6 _9 W2 v. z- M5 O, @复杂度量级排序:9 `. _, i* k7 E2 n. L
1 g: u- p5 C7 p7 l4 s: Y 8 W2 X& s0 h! C( @5 o7 l. i1. 常数阶O(1) ' b% T/ i4 Y/ i( _/ z: K3 a0 n( V4 r无论代码执行了多少行,其他区域不会影响到操作,这个代码的时间复杂度都是O(1) ! M. Y/ `8 d3 P W) H " T" {! w9 H" M2 C" K+ ]/ t T2 L/ t6 b; ^% ^void swapTwoInts(int &a, int &b) 3 w. v8 D/ d! C3 Z, Y2 \/ n4 |6 L6 B{* o2 ?& ]( F' |. K
int temp = a; 9 \, z" o( ^7 l0 ` a = b; 8 n% W+ {& }) C. j- b$ y8 ` b = temp;' W1 H1 ~/ S. C+ T, l" T6 ]9 _
}9 {: i5 M) x* F I1 ~- G
1 ) B2 d+ K% `- u4 K5 E$ E! O2 1 B" l: U/ R1 k7 _6 T3 B/ A Q3$ C$ r( o `9 `0 `! i$ f4 `3 m: t# P
4% l3 Z" q2 u5 [
5 8 S& f; s$ r( _/ L' F; T0 r5 N* a6# I$ g. i B) n8 m% ^' m9 v
2. 线性阶O(n)$ Z' O/ i5 R1 x8 D
8 o4 h6 v- n8 |1 k在下面这段代码,for循环里面的代码会执行 n 遍,因此它消耗的时间是随着 n 的变化而变化的,因此可以用O(n)来表示它的时间复杂度。/ Q% N2 V/ [: Y+ w5 e4 I
0 C; C; p8 W' dint sum ( int n ) 0 Z x( v/ P0 H* t{ 8 x/ D7 I, V% s5 e/ p5 A int ret = 0;2 M( q, |5 m* K" v/ _5 O8 q
for ( int i = 0 ; i <= n ; i ++)# [" a: Y' u' O+ j: e
{! C _; T# c- o5 ?# ?: T# Z
ret += i;1 p6 C& `3 V) _. e
} " Z9 X$ }8 B) N" c9 k7 e return ret;$ L/ s( M4 A$ p
}0 @- c0 {1 E5 m
1% l1 \( f" s9 f# A2 I
2 ; q" U0 {0 Q( F" ~5 Z3# _$ x9 j2 Y; o" g8 n1 R$ U
4" V h2 r7 V9 A2 ~! M9 o6 {4 e. |+ p/ l
5 . G' {) T _! c W5 l3 k60 P/ P3 ^9 ^0 D% i, r A
7 7 v4 ^4 L) n/ k( C7 y& c2 P* _+ T8+ X* d: B/ l) u, n
9 3 b4 _+ A0 T1 e4 L4 z: I3. 平方阶O(n2)9 [2 p% S' k0 G: j. @9 {- Q( B6 \
当存在双重循环的时候,即把 O(n) 的代码再嵌套循环一遍,它的时间复杂度就是 O(n2) 了。8 [" Q, e0 P. C+ C4 R: N
% ^/ M" i6 k4 l, j' j7 ^; i1 Y+ }
' G7 u+ ^8 s- L$ K' ovoid selectionSort(int arr[],int n){ ! i& F# p; |, ^. P! z" N ]7 U, e' d for(int i = 0; i < n ; i++){* s; D8 k+ w9 k, D
int minIndex = i; - [4 N$ U# R/ H for (int j = i + 1; j < n ; j++ ) / \0 v o5 p3 \* P if (arr[j] < arr[minIndex]) / Q, q- u& l/ |* [( D! R r minIndex = j;- a* K% k. A/ Y# a" g+ ]
swap ( arr, arr[minIndex]);+ ]$ i1 w$ S8 y/ |& l; N
} * K2 q8 U, ^# D' k) h' \0 C } # X, Q5 w6 N" L5 Q; h4 G11 A ]) j) S0 @1 P
2- V5 X% M3 Z! N/ ?9 Z* u4 {8 u
3 7 |% P0 R* g0 b# D/ m4 8 p4 g; ]$ o! I N$ a1 |. L5- I) ]$ R( y+ I, F6 C
6, w6 c$ ?+ w! Z Z+ x* W
7 1 ~* W" t1 E/ u, X, i2 A" a( a8 F8 : W6 x V6 v& y$ W" e6 Z6 t9 - Q; r" Q* ^4 J+ U9 D7 |3 \# t这里简单的推导一下7 [ F( G) h# K, e* l! E
& `( B0 u8 g& F: L, y当 i = 0 时,第二重循环需要运行 (n - 1) 次 ) ]" @' O) ?! N! x, G % k8 L+ {3 E. p当 i = 1 时,第二重循环需要运行 (n - 2) 次 - d$ J1 _$ d6 g) B9 D" X# b# t
。。。。。。 5 Z3 E" c* K' U, y3 f9 X8 h ( E- b, y6 v" p: Y0 R不难得到公式: 9 b7 ^9 v- |- J. z1 r1 S6 x Q6 u* K$ _ I
(n - 1) + (n - 2) + (n - 3) + ... + 04 k s- H6 W/ m8 ^/ ?
= (0 + n - 1) * n / 2 : @4 z' S3 y# E1 `3 |0 X6 ^/ Q$ L= O (n ^2) [7 J( a2 k) e! e }# [: \. N! Z
18 f0 @! W: y2 G$ @
2 # S6 j8 m% ~7 v3 _3 v3( K! O1 A( v/ {% O# ^" O
4. 对数阶O(logn)+ i/ d5 n! e/ J. ~
1 k/ ~5 Q4 F0 E: L/ W+ h( H6 E" h% j
int binarySearch( int arr[], int n , int target){ , i! p( F0 E) z4 y' { int l = 0, r = n - 1;' d* K( g: h# r0 ^; K! K
while ( l <= r) {: z$ A- R* g7 V+ o2 z$ c, _
int mid = l + (r - l) / 2; ) z- l3 T& p) O' i1 m! H2 X if (arr[mid] == target) return mid;6 {) ?) R9 a8 e c6 o
if (arr[mid] > target ) r = mid - 1;- l. ^" h5 ~. V H" N/ z
else l = mid + 1; 3 Z5 G9 U( ^) ^$ B6 Y% N } & ~6 e" b9 I& X1 ^0 ]3 o# s! g return -1; & K$ B# J6 j* V8 V+ i7 s- y0 O} 6 P1 v$ a! i, T& \1$ y& @) p, u P4 V7 [/ d
2 . F6 K8 q$ C: W+ K& [33 e" ]5 w7 J5 O' b
4; z+ L% {2 f6 V9 z4 N. R
5 4 ~ k# D6 l J: e' a60 X9 n0 d2 P' P# N
7 8 h( S* i D( q- U. s# X9 g: ~8 6 u! \( C9 h5 Y$ Q+ m" t9 * Y7 O" H6 y! }( H10 2 O) R! x+ Y7 w在二分查找法的代码中,通过while循环,成 2 倍数的缩减搜索范围,也就是说需要经过 log2^n 次即可跳出循环。 7 A8 G2 d7 }* @# V( I% v) h8 k2 z; X, N; ]* { i7 `
五、不常见的时间复杂度 / B1 y) ]. A$ V/ z3 ]6 A# T1. 最好情况时间复杂度(best case time complexity) 8 a% D3 H6 n3 q# l: Q最好情况时间复杂度就是,在最理想的情况下,执行这段代码的时间复杂度。在最理想的情况下,要查找的变量 x 正好是数组的第一个元素,这个时候对应的时间复杂度就是最好情况时间复杂度。 4 r$ q9 p# C) `" f3 {4 C# m9 `; {2 Y8 B# j8 V9 S
2. 最坏情况时间复杂度(worst case time complexity)5 K: ?& |) w4 ]0 x& s
最坏情况时间复杂度就是,在最糟糕的情况下,执行这段代码的时间复杂度。如果数组中没有要查找的变量 x,我们需要把整个数组都遍历一遍才行,所以这种最糟糕情况下对应的时间复杂度就是最坏情况时间复杂度。 % x9 S: v) D- q6 L 2 |) p# [4 p b9 s9 x! b最好、最坏情况时间复杂度指的是特殊情况下的时间复杂度。" j2 P/ `8 s. C- x4 P3 n* i, K
动图表明的是在数组 array 中寻找变量 x 第一次出现的位置,若没有找到,则返回 -1;否则返回位置下标。 ) e0 R! d9 k. l$ @4 N+ w* s& j + G' F+ J9 b5 I s6 `" z( s, Dint find(int[] array, int n, int x) {2 H2 H. x! W4 g' R7 W
for ( int i = 0 ; i < n; i++) { ) y. p m$ ~; B) S if (array == x) {% \; C# q- P( z6 N: |, i9 x8 {$ U
return i;' ]1 e% }# ]2 O9 q+ a9 d2 u
break;" d/ J& j9 J) \1 ?! Y1 w, |
} , V; b/ O1 g& {" y: K }7 y' p5 a: }$ D P
return -1; * O t+ X" J$ b6 r8 q( o& Z ?}% Z$ q' r/ r& i* S3 y+ @! ~3 Q6 x% G
1 1 q6 {: z" H+ O. {- C2& t& Y: w* F6 x, n# b* o
3 & Z1 y" m! s0 U5 C* x4 Q' W0 i" Z, u* A6 H) ~
5 ' @ a1 E% [9 K$ d+ Q. K6 : O: H5 v, u7 s) N7 ; J; V9 [0 B( g: m$ ]7 N- Q2 G4 x" Z8 1 b% Z0 {9 B9 Q4 b9 . w$ D; P+ |9 N- [* a在这里当数组中第一个元素就是要找的 x 时,时间复杂度是 O(1);而当最后一个元素才是 x 时,时间复杂度则是 O(n)。 ; P7 @9 X! @0 j/ V2 W" |% R5 N0 G: k- @4 J
最好情况时间复杂度就是在最理想情况下执行代码的时间复杂度,它的时间是最短的;最坏情况时间复杂度就是在最糟糕情况下执行代码的时间复杂度,它的时间是最长的。' m* Y+ H- ?/ Z4 j, g6 E: [3 i! q
3 l! ? h/ k/ {: j* U O
3. 平均情况时间复杂度(average case time complexity)/ F* n' V! O- r9 x1 ~3 C, `: x, S
最好、最坏时间复杂度反应的是极端条件下的复杂度,发生的概率不大,不能代表平均水平。那么为了更好的表示平均情况下的算法复杂度,就需要引入平均时间复杂度。) l$ Z/ V$ E( U! K
/ J/ L: K! n* }
平均情况时间复杂度可用代码在所有可能情况下执行次数的加权平均值表示。 ) t5 Z- t, P3 {. N6 z( }% c8 C7 }. l3 V" l
还是以 find 函数为例,从概率的角度看, x 在数组中每一个位置的可能性是相同的,为 1 / n。那么,那么平均情况时间复杂度就可以用下面的方式计算: & v9 \5 H( y. D. D. E$ V5 h * D/ u9 e8 I0 i' w. b) S. [((1 + 2 + … + n) / n + n) / 2 = (3n + 1) / 4 0 G/ e5 [/ j) n& Y+ g+ k1" a- W( `/ x# I" \( W8 p! d5 T
- Y1 V) a& y. ]
find 函数的平均时间复杂度为 O(n)。 \! ?6 o4 Q" L" x8 W- ~ ! T( L/ d8 C% v4. 均摊时间复杂度(amortized time complexity)8 Y8 |( a6 ~6 @& _
我们通过一个动态数组的 push_back 操作来理解 均摊复杂度。 7 ^+ l( p% m* U% U3 f7 e' h. f& J0 E+ J
) L4 K) z1 x) k/ p& R: a9 dtemplate <typename T> ! J3 t7 ?$ Z5 m# m. I class MyVector{ " c7 F& l& V( z+ j private:/ P# J' ]1 O# z
T* data;; z) ]1 N+ G/ }* B8 v" |4 b5 \7 K' V
int size; // 存储数组中的元素个数4 t& |/ w/ Z6 w8 U; X2 a) J
int capacity; // 存储数组中可以容纳的最大的元素个数- o: |$ ?1 d8 q4 l3 b
// 复杂度为 O(n)# l, U$ ]. T( t! W
void resize(int newCapacity){ % B% I- Q5 I" T2 L1 e; y% J! K; G T *newData = new T[newCapacity];8 c9 @' U8 W% {6 d6 }7 t
for( int i = 0 ; i < size ; i ++ ){ / c3 |- |4 C5 J! C' ?* I0 r* y newData = data;0 P/ w8 D5 M ^$ S4 k. A- d
}5 d, J, }' C# I
data = newData; * x/ b0 D6 Z1 u3 q5 W# T+ w capacity = newCapacity; * m+ G5 f7 _# p- Z }+ E' U1 ^ ~1 O1 h
public:6 `0 A( \4 w( H
MyVector(){4 ]" o) j8 `: y6 j0 i
data = new T[100];& q/ N7 D& \0 ^2 ]: p- c+ v
size = 0; 1 U3 b& W, I* b, D: O capacity = 100;" o' z# m) a6 R, p
} ' z. s- z% S& I- N // 平均复杂度为 O(1)* {5 {' \; _, M( ^
void push_back(T e){5 e0 p8 I: F: d) l
if(size == capacity)" a$ u. a" C) e2 K3 L' q& \6 r2 }
resize(2 * capacity);: |( V$ G, f; d+ Q# D: A9 N' f Q
data[size++] = e; : R2 a- N+ M7 M5 R } X1 r* r: _9 Z* Z2 {# y* } // 平均复杂度为 O(1)* i L6 q- M# a* z g' ], t
T pop_back(){0 |1 K9 }; n/ }
size --; + @2 Q9 Y' {0 [8 c1 F# H0 p return data[size];3 Z: S7 j5 P' j; k* i- d
} & @- g; M& C8 `& y* n) R8 ~! _# |3 Y0 X" i1 s; ?( j
}; ; j3 J' }% s! \2 v' ^, H 7 c+ K9 S( k x) F4 g7 W, n3 K! `. O U- w
1 9 W, O* ?5 k2 ~3 m# R8 l4 `2& d/ T/ s5 j: l8 }
3" l6 C, ?" K1 n* R C
4 3 f! ~$ b" J& ~% s+ n! v4 H) \5" |( R& a% t8 E: f
6* S; U5 P! v* V" @ d( l
7 3 u S7 e" O9 y+ ~' C# e8 . d( C- K3 v$ |$ ?* z9 % `; q1 N. H4 `' V) j; e: `/ `! n# t10 I6 |8 ]0 B2 d# [. Q
118 @" m3 z3 B% b& u
12 % u7 u a9 B* z! G1 H* o13 ( R3 t5 m/ G, D1 v2 u/ T149 \$ b' v$ \2 Y0 \/ A
15 ; }8 d- [' `" I2 s, K9 T16 |6 E- U" V5 \6 P; p R
17 : y& u: {3 s6 J6 q! G) H$ p0 H* j18& p, {6 N4 `4 P$ n( ^9 r+ B9 g2 D# V
196 Q7 U, Y! L/ x i, L5 r
205 F# v2 z( c- I- H
21. `: a( L3 _% f( {! ?( I+ C
22! X: O3 R7 j# Y, U
23 4 ^" J& V7 r0 T24 7 i2 B7 l( g Z9 F) D5 z25 " W4 Q8 b+ I6 x% F8 `( u26 1 G; S$ Z! } w/ r! Q8 @27 7 s' j1 b% j# b1 `- }' ]( Y28. w" u- h* W' n, l
299 F6 x2 D9 I7 r' R( Z# N+ n. i' O
303 m! Z7 D+ q. S E8 F
31 # ] D, ?1 b# H2 Q7 M* L/ s32 ( z( {3 y* i$ O; X+ R1 N0 p T33- Q% g$ P" y" L L& ^) H
347 P# X( D! q# U. P- [6 z2 m
35 - k, N! F' |* t7 h! ~push_back实现的功能是往数组的末尾增加一个元素,如果数组没有满,直接往后面插入元素;如果数组满了,即 size == capacity ,则将数组扩容一倍,然后再插入元素。5 |7 Q8 W, I! r6 O
6 C, E: b- Y, j+ q
例如,数组长度为 n,则前 n 次调用 push_back 复杂度都为 O(1) 级别;在第 n + 1 次则需要先进行 n 次元素转移操作,然后再进行 1 次插入操作,复杂度为 O(n)。8 ?1 }6 a# _2 Q: }) v
x5 b% V) X4 v }3 J' N$ B, [因此,平均来看:对于容量为 n 的动态数组,前面添加元素需要消耗了 1 * n 的时间,扩容操作消耗 n 时间 , ) k: ?7 M) c: r5 B总共就是 2 * n 的时间,因此均摊时间复杂度为 O(2n / n) = O(2),也就是 O(1) 级别了。& r4 V4 ^1 F7 f" [* L
( e& H# h3 t( d! V& O: k% v可以得出一个比较有意思的结论:一个相对比较耗时的操作,如果能保证它不会每次都被触发,那么这个相对比较耗时的操作,它所相应的时间是可以分摊到其它的操作中来的。% v p* D M; Z, g# m
( v( V1 D7 r1 b! I% f X
六、如何掌握好复杂度分析方法?" k, S8 m% z; K
复杂度分析关键在于多练,所谓孰能生巧。7 e: z+ X( J1 Y" W
e+ r# S7 P7 @0 ?- ~+ S1. 大O标记 6 Q7 w* l* M3 V: G2 C/ H* T7 A. T从 CPU 的角度来看,这段代码的每一行都执行着类似的操作:读数据-运算-写数据。尽管每行代码对应的 CPU 执行的个数、执行的时间都不一样,但是,我们这里只是粗略估计,所以可以假设每行代码执行的时间都一样,为 单位时间(unit_time)。$ d! T2 \& m7 P3 A/ @: `$ m* _
H$ s3 x* U7 d2 v, |# m
算法运行工作量(基本操作重复执行的次数总和)的大小是数据规模 n 的函数,记作:f ( n ) 。 $ w3 ?0 Q" j1 L% M! j5 x ! E$ Q m" u8 o! W- j7 y( h/ S则代码执行时间可表示为: ) H$ e! D! J1 t/ T " a# r2 d% v1 qT(n)=O(f(n)). A5 O- @8 ^ H( s
1 & C z& K0 t& u; ^4 y% o大 O 时间复杂度实际上并不具体表示代码真正的执行时间,而是表示代码执行时间随数据规模增长的变化趋势,O表示算法执行时间的增长率和 f ( n ) 的增长率相同,称为渐进时间复杂度,简称时间复杂度。) P) f, J1 a# y
2 F% m6 n3 l5 X+ H+ r时间复杂度的基本原则:, w2 `$ O. F7 S0 Q. v O- i4 z4 l