【原创连载】算法素颜(第3篇):KO掉大O不再是菜鸟的梦——时间复杂度

创作时间:2019-03-23

作者:周林

版权所有,转载请联系作者获取授权、并注明作者与出处

周林的简书主页

周林的CSDN博客主页

周林的知乎主页

                              


 

     在本系列第1篇《走下神坛吧!算法》中提到了:计算复杂度分为时间复杂度与空间复杂度。本篇文章来讲讲时间复杂度。


如何度量时间复杂度

       时间复杂度由所消耗的时间决定。所消耗的时间由硬件与软件共同决定。在同一硬件条件下,所消耗的时间由软件决定。

       通常意义上的算法指的是软件算法,所以在谈论时间复杂度时,聚焦软件的时间开销。

       软件的时间开销 = 软件各组成部分的时间开销之和。

       软件最基本的组成部分是语句(或者微观意义下的指令)。

       注:因为在同一硬件条件、特定的编程语言环境下,基本语句由多少条指令构成、运行的模式都是固定的,所以直接以语句作为基本考察单位即可。


整体等于各部分之和

      对大部分编程语言而言,基本语句无外乎以下两种:

      1. 读写语句

      2. 比较语句

      注:条件语句(if-else)、循环语句(while/for)在这里都不算基本语句,而被看作复合语句。

             原因如下:

             对于条件语句,一般形式如下:

             if (xxx) {  // 这条语句相当于先运行比较语句(比较xxx与true的值),

                         //  然后根据比较结果,运行aaa(aaa可以是一个简单语句,也可以是复合语句)

                   aaa;

             } else if (yyy) {   // 这条语句相当于先运行比较语句(比较yyy与true的值),

                        //  然后根据比较结果,运行bbb(bbb可以是一个简单语句,也可以是复合语句)

                  bbb;

             }


            对于循环语句,一般形式如下:(为简单起见,这里只做while的示例,其他循环类型依此来推)

            while (xxx) {      // 这条语句相当于先运行比较语句(比较xxx与true的值),

                                     //  然后根据比较结果,运行bbb(aaa可以是一个简单语句,也可以是复合语句)

                                     // 再次运行比较语句(比较xxx与true的值)判断是否出循环体

              aaa;

            }

    将基本语句的时间开销看作“单位1”,那么整个程序的时间开销就是所有这些“单位1”之和。

     推论3.1:

             假设程序结构L由n个子结构Li组成,即:

​            L = \sum_{i=1}^{n}{Li}

             那么运行L的整体时间开销T(L)为:

           T(L) = T(\sum_{i=1}^{n}{Li})=\sum_{i=1}^{n}{T(Li)}


T(L) = f(n) 

     来看几个例子:

             int i = 10;            // 赋值语句,属于基本语句,所以时间开销 = 1

             a > b;                 // 比较语句,数据基本语句,所以时间开销 = 1

             if (a > b) {          // 条件语句,属于复合语句。它包含了一条比较语句和一条赋值语句,所以时间开销 = 1 + 1 = 2

               c= 9; 

             }

       再来看一个复杂一点的:

             int buf_size = input_array.size;

             for (i = 1; i < buf_size; i++) {

               p = q + 1;

               if (p > i) {

                 break;

               }

             }

     整个程序结构L由一条赋值语句(记为J)和一个循环体(记为K)组成,则:L = J + K  (式3.1)

     循环体K由一条赋值语句X和一个条件语句Y组成,循环次数n在最坏情况下等于buf_size。所以在最坏情况下:

     K = n(X + Y)   (式3.2)

     代入式3.1,得到:L = J + n(X+Y) (式3.3)

     根据推论3.1,从式3.3可得到:

     T(L) = T(J) + T(n(X + Y)) = T(J) + nT(X + Y) = T(J) + n(T(X) + T(Y))  (式3.4)

     因为J, X为赋值语句,属于基本语句,所以T(J) = T (X) = 1

     因为Y为简单条件语句,T(Y) = 2

     如果buf_size = 10,那么n = 10

     将T(J), T(X), T(Y), n代入式(3.4)得到: T(L) = 1 + 10 x (1+2) = 31

     从上面的例子可以看出:

     (i) n不同,T(L)的值就不同;

     (ii) n越大,T(L)的值也越大;

     (iii) n等于buf_size,后者是由输入input_array的规模决定的

所以,可以看出:T(L)与输入规模相关。用数学的语言来描述就是:整体时间开销是一个关于输入规模因子的函数。

     用数学表达式表示就是:T(L) = f(n)


T= f(n)的现实意义是什么

     f(n)既然是个函数,那么由高中数学知识可知,f(n)的形式有很多种,如:

     ​f(n) = C

    f(n) = n^{2}

    f(n) = log _2n

    f(n) = a^n

    f(n) = a_{1}n^{k} + a_{2}n^{k-1} + a_{3}n^{k-2} + ... + a_{k-1}n^{2} + a_{k}n + b

     不同形式的f(n),在n增加的时候,T增加的速度不同。

     每一种f,对应一种算法,所以:

     推论3.2:

            T表示在同等输入规模(即n一定)的情况下,不同算法的时间开销。

            T的增加速度越快,对应算法的时间开销越大,也就表示该算法越复杂

“两种算法的复杂度在一个量级”到底是个什么鬼

       考虑T1 = 2n 和 T2 = n。

       很显然,在n固定的情况下,T1/T2 = 2。即:同等输入规模下,第一种算法的时间开销是第二种算法时间开销的2倍。

       这种复杂度关系总是常数倍的,即使n取无穷大也是。用数学语言表示就是:

​       \lim_{n \rightarrow \infty}{\frac{T_{1}}T_{2}} = C

       从而,我们得到一个关键结论:

       推论3.3:

             两种算法的复杂度在同一量级等价于      

                      \lim_{n \rightarrow \infty}{\frac{T_{1}}T_{2}} = C(其中C为常数)

       聪明的你肯定会问到:如果n趋向无穷大的时候,T1/T2不是常数呢?如果不是常数,那么通常是一个关于n的表达式,随着n的增加,值也增加。这表示T1/T2也将趋向无穷大。用数学术语表达就是“发散”。

       我们称这种况下,两种算法不在同一复杂度量级。

       推论3.4:

             算法1比算法2的复杂度量级高等价于

​             \lim_{n \rightarrow \infty}{\frac{T_{1}}T_{2}} = \infty


大O登场

         通常比较算法复杂度,只用比较量级即可。量级用O()表示。

         O()定义:

         (i) 如果算法T1与算法T2的复杂度在同一量级,那么O(T1) = O(T2)

         (ii) 如果算法T1比算法T2的复杂度量级高,那么O(T1) > O(T2)

         (iii) 如果算法T1比算法T2的复杂度量级低,那么O(T1) < O(T2)

         我们来看一个例子:

         T_{1} = a_{1}n^{k} + a_{2}n^{k-1} + a_{3}n^{k-2} + ... + a_{k-1}n^{2} + a_{k}n + b

          T_{2}= n^{k}

        比较T1与T2的复杂度量级:

​          \lim_{n \rightarrow \infty}{\frac{ a_{1}n^{k} + a_{2}n^{k-1} + a_{3}n^{k-2} + ... + a_{k-1}n^{2} + a_{k}n + b}{n^{k}}} = a_{1}

        根据推论3.3得到:T1与T2处于同一复杂度量级。

        根据上述O()的定义:O(T_{1}) = O(T_{2})

        这里其实蕴含了一个非常实用的结论:

        推论3.5:

               算法复杂度的大O表示可以简化为该算法最高阶部分的复杂度的大O表示。

        根据高等数学知识,我们可以得到:

​        O(a^n) > O(n^k) > O(nlog_2n) > O(n) > O(log _2n) > O(1)


写在最后的话

大O表示法最早由德国数论学家保罗·巴赫曼(英语:Paul Bachmann)在其1892年的著作《解析数论》(Analytische Zahlentheorie)首先引入的。而这个记号则是在另一位德国数论学家艾德蒙·朗道(英语:Edmund Landau)的著作中才推广的,因此它有时又称为朗道符号(Landau symbols)。

      大部分的算法或者复杂度理论的书籍,在介绍大O时,要么过于数学形式化,要么过于感性非严格化。本篇文章旨在用最少的数学知识、启发式行文方式、全新的原创视角,为读者构建一个清晰、严格的时间复杂度概念。

      相信看到这里,以后再面对时间复杂度和大O,你一定信心满满了:)      


本原创系列同步在以下自媒体上更新,敬请关注:

CSDN: https://blog.csdn.net/jintianyishiyeai/

知乎:https://www.zhihu.com/people/cubiezhou2017

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容