渐进时间复杂度分析

公众号:原与译

我们先来看一到算法题

给定一个自然数 n,然后求出前 n 个自然数的和 sum。( n > 0 )

如:</br>
n = 3,则 sum = 1 + 2 + 3 = 6</br>
n = 5,则 sum = 1 + 2 + 3 + 4 + 5 = 15

然后给出如下三种解法。

方法一

private int fun1(int n) {
  return n * (n - 1) / 2;
}

方法二

private static int fun2(int n) {
    int sum = 0;
    for (int i = 1; i <= n; i++) {
        sum = sum + i;
    }
    return sum;
}

方法三

private static int fun3(int n) {
    int sum = 0;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= i; j++) {
            sum++;
        }
    }
    return sum;
}

上述给出的三种方法都可满足给定的需求,但是如果需要在这三个方法中选出一个最优解,那就需要使用时间复杂度来衡量他们了。

如果某些行代码执行的时间不受用户输入影响,将该代码的执行时间记录为 常数时间,用 C 表示

在 fun1 中,方法体就一行代码。且该行代码的执行时间不受用户输入影响,即使入参 n 的值为 1000,10000.... fun1 的执行时间依然不变。所以可以把 fun1 的时间复杂度记为 C。

在 fun2 中,是有一些局部变量和一个 for 循环组成,int sum = 0; int i = 0; return sum; 这些代码并不会受输入参数 n 的影响,所以我们记录为 3C。而 for 循环中的条件判断,和 for 循环中的方法体都会执行 n 次(受入参 n 的影响),这部分的时间记录为 2n。所以 fun2 最后的时间复杂度为 3C + 2n。

在 fun3 是有一些局部变量和两个 for 循环构成,且这两个 for 循环是嵌套关系。外层 for 循环的时间复杂度为 n,内存 for 循环加上循环体的执行的之间复杂度为 2n。对于这中嵌套关系的循环,复杂度之间是相乘的。及 n * 2n = 2n²。再加上 3 个变量的声明的事件复杂度 3C。所以 fun3 的渐进时间复杂度为 2n² + 3C。

通常,在评估时间复杂度的时候,我们会忽略掉两个部分,一个部分是低阶,另一个部分是 常数阶。因为一般在评估时间复杂度的时候都是评估的最坏的时间复杂度,所以这两部分可以直接忽略掉。对应到上述三个方法中:

fun1 = C = O(1)
fun2 = 3C + 2n = O(n)
fun3 = 2n² + 3C = O(n²)

如果 for 循环中是简单的线性增加(减少)且其上限不是一个常数的情况下,那么可以认为当前的循环的时间复杂度为 O(n)

public void method1(int n){
    for(int i = 0; i < n; i++){
        // do O(1)
    }
} 

上述方法 method1 中的 for 循环的时间复杂就是 O(n)。

public void method2(int n) {
    for(int i = 0; i < 10000; i++) {
        // do O(1)
    }
}

此时,method2 方法中也有一个 for 循环,但是这个循环的执行时间并不受输入参数 n 的影响,所以,method2 的时间复杂度为 O(1)。

知道了上述三个方法的渐进时间复杂度,下面来看看他们的增长顺序:

假设有 f(n) = 2n² + n + 6 ; g(n) = 100n + 3,那么忽略完它们的低阶及常数阶之后则 f(n) = n²,g(n) = n。
由此我们可以判断 f(n) 的增长顺序是要高于 g(n) 的。也就是 f(n) 的时间复杂度比 g(n) 要高。

下面是上述是三种方法各自时间复杂度的图示:

横坐标 n 表示用户输入的数据量
纵坐标 t 表示计算消耗的时间

从图可以看出,增长顺序由低到高为 O(1) < O(n) < O(n²)。算法的性能排序是 fun1 > fun2 > fun3。

几种表示复杂度的渐进符号
我们在上面表示算法的复杂度时候用的渐进符号是大 O 符号,其实还有两种渐进符号表示复杂度的,它们分别是 Θ 和 Ω 符号 。

O 渐进符号定义了一个算法的上限,也就是我们说的最坏时间复杂度。例如,插入排序,最佳的情况下的时间复杂度是 O(n),而最坏的情况是 O(n²),所以我们可以说插入排序的事件复杂度是 O(n²)

Θ 渐进符号表示的是一个算法从上线到下限的时间复杂度。一个简单的方法获取 Θ 渐进表达式为去掉低阶以及忽略常数阶。例如:3n³ + 6n² + 6000 = Θ(n³),而如果使用Θ表示插入排序的事件复杂度,则需要如下:

  1. 插入排序的最坏时间复杂度为 Θ(n²)
  2. 插入排序的最佳时间复杂度为 Θ(n)

Ω 渐进符号表示一个算法的下限,也就是我们说的最佳时间复杂度。如果我们需要就拿一个算法的最佳时间复杂度的时候,Ω会被用到。Ω 符号是这三个渐进符号中使用最少的。


下面我们来分下各种循环的时间复杂度。

O(1): 如果一个方法或者语句中不包含循环,递归,或者没有调用其他时间复杂度为非常数的情况下,这个方法的时间复杂度被认为是 O(1)。 注意,一个循环或者递归如果执行的次数为常数级。则它们的时间复杂度依然被认为是 O(1),如下代码中的循环就是 O(1):

int c = 100;
for (int i = 1; i <= c; i++){
    // 执行 O(1) 的循环体
}

O(n): 如果循环是以一个恒定的值递增/递减,则循环的时间复杂度就可以记为 O(n)

c 是一个常数;n 为用户输入

for (int i = 0; i < n; i = i+c){
    // do O(1)
}

上述 for 循环中,n 是用户输入,c 是一个常量,我们假设 n = 10; c = 2; 变量 i 的值依次递增为:0,2,4,6,8;执行次数为 n/c;记为 O(n/c),最后忽略掉常数阶c,则为 O(n)。

O(n²):如果一个方法中包含两个嵌套关系的循环,且循环里面的循环变量都是以一个恒定的值递增/递减。那么这个方法的时间复杂度可以记为 O(n²),如下:

// n 表示用户输入;c 表示常数
for (int i = 1; i <= n; i += c) {
    for (int k = 1; k <= n; k += c) {
        // do O(1)
    }
}

O(Logn): 如果循环中的循环变量除以/乘以 一个常数,那么该循环的事件复杂度可以记为 O(Logn)

// n 表示用户输入;c 表示常数
for (int i = 0; i <= n; i *= c) {
    // do O(1)
}

我们假设 n = 32; c = 2;则循环变量 i 的值依次为:0,2,4,8,16,32 ......

我们可以把 i 的值记为 1 , C , C² , C³ ...... C^k-1

接着就是 C^k-1 <= n,我们求出 k 的值,这里是一个对数运算,算出来就是 k < Logc^n+1 (Log 以 C 为底 n+1 的对数),接着我们忽略掉 Logc^n+1 中的常数阶 c 和 1。最终的结果就是 Logn。

对数计算小常识:
假设 a^x = N ,我们知道 a = 2,x = 3, 那么 N 就等于 8,这是一个指数运算,求的是 N
对数运算就是已知 a 和 N 求 x,如 2^x = 8, 我们一眼就能看出 x = 3。写作 x = Log2^8 (Log 以 2 为底 8 的对数)

例如,二分查找的时间复杂度就是 O(Logn)

O(LogLogn): 如果一个 for 循环中的循环变按照指数级的递增或者递减,那么,这个循环的事件负责度可以记为 O(LogLogn)。

// n 表示用户输入;c 表示常数
// pow 表示 i^c
for (int i = 2; i < n; i = Math.pow(i, c)) {
    // do O(1)
}

上述代码中,循环变量 i 的初始值为 2,i 的值变化如下及循环的时间复杂度计算如下:

我们最终忽略低阶及常数阶之后得到的渐进时间复杂度为 O(LogLogn)

O(nLogn): 下面我们来看一个渐进时间复杂度为 O(nLogn) 的例子。

public void fun(int n) {
    for (int i = 0; i < n; i++ ){
        for (int k = 1; k < n; k = k*2){
            // do O(1)
        }
    }
}

上述方法 fun 中,包含了两个嵌套的 for 循环,外层 for 循环的事件复杂度为 O(n); 内层 for 循环的事件复杂度为 O(logn)。
**
在计算嵌套 for 循环的时候,需要将每一层的时间复杂度相乘。则有:O(n) * O(logn),所以上述方法 fun 的渐进时间复杂度为O(nlogn)。

下面看几个例子

public void demo1(int n) {
    for (int i = 0; i < n; i++) {
        // do O(1)
    }
    
    for (int i = 1; i < n; i = i * 2) {
        // do O(1)
    }
    
    for (int i = 0; i < 100; i++) {
        // do O(1)
    }
}

上述方法 demo1 中,包含了三个并列的 for 循环,现在我们很容易就能判断出第一个 for 循环的时间复杂度为 O(n); 第二个 for 循环的时间复杂度为 O(logn); 而第三个 for 循环的时间复杂度为 O(1), 因为它的执行次数并不受用户输入的影响。

对于并列的 for 循环,我们需要把每个 for 循环的时间复杂度相加,所以方法 demo1的事件复杂度为:O(n) + O(logn) + O(1) 。

那么我们忽略掉低阶 O(logn) 和常数阶 O(1),最后得出方法 demo1的时间复杂度为 O(n)。

public void demo2(int n) {
    for (int i = 0; i < n; i++) {
        for (int j = 1; j < n; j = j * 2){
            // do O(1)
        }
    }
    
    for (int i = 0; i < n; i++) {
        for(int j = 1; j < n; j++) {
            // do O(1)
        }
    }
}

分析:demo2 中,有两个并列型大的 for 循环,每一个 for 循环都是一个嵌套型的循环。第一个大的 for 循环的时间复杂度为 O(nlogn);第二个大的 for 循环的时间复杂度为 O(n²)。demo2 的时间复杂度为 f(n) = O(nlogn) + O(n²) 。省略低阶 O(nlogn) 之后,demo2 的渐进时间复杂度为 O(n²)。

注意:对于一个方法中包含多个 for 循环的情况,如果它们是并列的,则该方法的时间复杂度为多个 for 循环的时间复杂度的和;
如果是嵌套的,则该方法的时间复杂度为多个 for 循环的时间复杂度的乘积。

下面再看一个例子

// c 表示一个常数
public void demo3(int m, int n) {
    for(int i = 1; i <= m; i += c) {
        // do O(1)
    }
    
    for(int i = 1; i <= n; i += c) {
        // do O(1)
    }
}

对于 demo3 中,它的时间复杂度为 f(n) = O(m) + O(n) = O(m + n);如果 m == n,则 f(n) = O(2n) = O(n)。

好了,上面分别讲述了 O(1), O(n), O(n²),O(logn), O(nlogn) 等情况,如有错误,还望指出。

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