漫画:什么是时间复杂度?

简介: 时间复杂度的意义,究竟什么是时间复杂度呢?让我们来想象一个场景:某一天,小灰和大黄同时加入了一个公司......一天过后,小灰和大黄各自交付了代码,两端代码实现的功能都差不多。大黄的代码运行一次要花100毫秒,内存占用5MB。小灰的代码运行一次要花100秒,内存占用500MB。

image.png

image.png


image.png


image.png

image.png

image.png



时间复杂度的意义


究竟什么是时间复杂度呢?让我们来想象一个场景:


某一天,小灰和大黄同时加入了一个公司......

image.png


一天过后,小灰和大黄各自交付了代码,两端代码实现的功能都差不多。


大黄的代码运行一次要花100毫秒,内存占用5MB


小灰的代码运行一次要花100秒,内存占用500MB


于是......

image.png

image.png





由此可见,衡量代码的好坏包括两个非常重要的指标:


1.运行时间

2.占用空间

image.png

image.png


基本操作执行次数


关于代码的基本操作执行次数,我们用四个生活中的场景来做一下比喻:


场景1. 给小灰一条长10寸的面包,小灰每3天吃掉1寸,那么吃掉整个面包需要几天?

image.png


答案自然是 3 X 10 = 30天。


如果面包的长度是 N 寸呢?


此时吃掉整个面包,需要 3 X n = 3n 天。


如果用一个函数来表达这个相对时间,可以记作 T(n) = 3n



场景2.  给小灰一条长16寸的面包,小灰每5天吃掉面包剩余长度的一半,第一次吃掉8寸,第二次吃掉4寸,第三次吃掉2寸......那么小灰把面包吃得只剩下1寸,需要多少天呢?


这个问题翻译一下,就是数字16不断地除以2,除几次以后的结果等于1?这里要涉及到数学当中的对数,以2位底,16的对数,可以简写为log16。


因此,把面包吃得只剩下1寸,需要 5 X log16 = 5 X 4 = 20 天。


如果面包的长度是 N 寸呢?


需要 5 X logn = 5logn天,记作 T(n) = 5logn



场景3.  给小灰一条长10寸的面包和一个鸡腿,小灰每2天吃掉一个鸡腿。那么小灰吃掉整个鸡腿需要多少天呢?

image.png


答案自然是2天。因为只说是吃掉鸡腿,和10寸的面包没有关系 


如果面包的长度是 N 寸呢?


无论面包有多长,吃掉鸡腿的时间仍然是2天,记作 T(n) = 2

场景4.  给小灰一条长10寸的面包,小灰吃掉第一个一寸需要1天时间,吃掉第二个一寸需要2天时间,吃掉第三个一寸需要3天时间.....每多吃一寸,所花的时间也多一天。那么小灰吃掉整个面包需要多少天呢?


答案是从1累加到10的总和,也就是55天。


如果面包的长度是 N 寸呢?


此时吃掉整个面包,需要 1+2+3+......+ n-1 + n = (1+n)*n/2 = 0.5n^2 + 0.5n。


记作 T(n) = 0.5n^2 + 0.5n

image.png





上面所讲的是吃东西所花费的相对时间,这一思想同样适用于对程序基本操作执行次数的统计。刚才的四个场景,分别对应了程序中最常见的四种执行方式:


场景1, T(n) = 3n,执行次数线性的。

  1. void eat1(int n){
  2.    for(int i=0; i<n; i++){;
  3.        System.out.println("等待一天");
  4.        System.out.println("等待一天");
  5.        System.out.println("吃一寸面包");
  6.    }
  7. }

vo


场景2, T(n) = 5logn,执行次数对数


  1. void eat2(int n){
  2.    for(int i=1; i<n; i*=2){
  3.        System.out.println("等待一天");
  4.        System.out.println("等待一天");
  5.        System.out.println("等待一天");
  6.        System.out.println("等待一天");
  7.        System.out.println("吃一半面包");
  8.    }
  9. }


场景3,T(n) = 2,执行次数常量的。


  1. void eat3(int n){
  2.    System.out.println("等待一天");
  3.    System.out.println("吃一个鸡腿");
  4. }


场景4,T(n) = 0.5n^2 + 0.5n,执行次数是一个多项式


  1. void eat4(int n){
  2.    for(int i=0; i<n; i++){
  3.        for(int j=0; j<i; j++){
  4.            System.out.println("等待一天");
  5.        }
  6.        System.out.println("吃一寸面包");
  7.    }
  8. }



渐进时间复杂度


有了基本操作执行次数的函数 T(n),是否就可以分析和比较一段代码的运行时间了呢?还是有一定的困难。


比如算法A的相对时间是T(n)= 100n,算法B的相对时间是T(n)= 5n^2,这两个到底谁的运行时间更长一些?这就要看n的取值了。


所以,这时候有了渐进时间复杂度(asymptotic time complectiy)的概念,官方的定义如下:


若存在函数 f(n),使得当n趋近于无穷大时,T(n)/ f(n)的极限值为不等于零的常数,则称 f(n)是T(n)的同数量级函数。


记作 T(n)= O(f(n)),称O(f(n)) 为算法的渐进时间复杂度,简称时间复杂度。


渐进时间复杂度用大写O来表示,所以也被称为大O表示法

image.png


image.png




如何推导出时间复杂度呢?有如下几个原则:

  1. 如果运行时间是常数量级,用常数1表示。
  2. 只保留时间函数中的最高阶项
  3. 如果最高阶项存在,则省去最高阶项前面的系数。



让我们回头看看刚才的四个场景。


场景1:

T(n) = 3n

最高阶项为3n,省去系数3,转化的时间复杂度为:

T(n) =  O(n)

image.png



场景2:

T(n) = 5logn

最高阶项为5logn,省去系数5,转化的时间复杂度为:

T(n) =  O(logn)

image.png



场景3:

T(n) = 2

只有常数量级,转化的时间复杂度为:

T(n) =  O(1)

image.png

场景4:

T(n) = 0.5n^2 + 0.5n

最高阶项为0.5n^2,省去系数0.5,转化的时间复杂度为:

T(n) =  O(n^2)

image.png


这四种时间复杂度究竟谁用时更长,谁节省时间呢?稍微思考一下就可以得出结论:


O(1)< O(logn)< O(n)< O(n^2)



在编程的世界中有着各种各样的算法,除了上述的四个场景,还有许多不同形式的时间复杂度,比如:

O(nlogn), O(n^3), O(m*n),O(2^n),O(n!)


今后遨游在代码的海洋里,我们会陆续遇到上述时间复杂度的算法。

image.png



时间复杂度的巨大差异


image.png


image.png




我们来举过一个栗子:


算法A的相对时间规模是T(n)= 100n,时间复杂度是O(n)

算法B的相对时间规模是T(n)= 5n^2,时间复杂度是O(n^2),


算法A运行在小灰家里的老旧电脑上,算法B运行在某台超级计算机上,运行速度是老旧电脑的100倍。


那么,随着输入规模 n 的增长,两种算法谁运行更快呢?


image.png


从表格中可以看出,当n的值很小的时候,算法A的运行用时要远大于算法B;当n的值达到1000左右,算法A和算法B的运行时间已经接近;当n的值越来越大,达到十万、百万时,算法A的优势开始显现,算法B则越来越慢,差距越来越明显。


这就是不同时间复杂度带来的差距。


image.png




几点补充:


小灰写这篇时间复杂度的科普时才意识到,对于基础概念的讲解,比讲解具体算法要困难得多。希望大家对本文多提出宝贵意见,感谢大家!




—————END—————

相关文章
|
7月前
|
算法
时间复杂度与空间复杂度(自漫画算法)
时间复杂度与空间复杂度(自漫画算法)
33 0
|
搜索推荐 算法
齐姐漫画:排序算法(一)
借用《算法导论》里的例子,就是我们打牌的时候,每新拿一张牌都会把它按顺序插入,这,其实就是插入排序。
152 0
齐姐漫画:排序算法(一)
|
搜索推荐 算法 IDE
齐姐漫画:排序算法(二)之「 归并排序」和「外排序」
那我们借用 cs50 里的例子,比如要把一摞卷子排好序,那用并归排序的思想是怎么做的呢?
178 0
齐姐漫画:排序算法(二)之「 归并排序」和「外排序」
|
算法
漫画:什么是二分查找?(修订版)
如果我们把场景转换成最初的面试问题:在包含1000个整型元素的有序数组中查找某个特定整数,又该如何去做呢?
124 0
漫画:什么是二分查找?(修订版)
|
存储 算法
漫画:什么是归并排序?
举个例子,有A、B、C、D、E、F、G、H一共8个武术家参考参加比武大会。 第一轮,两两一组,有4名选手胜出(四分之一决赛) 第二轮,两两一组,有两名选手胜出(半决赛) 第三轮,仅剩的两人一组,冠军胜出(总决赛)
122 0
漫画:什么是归并排序?
|
存储 缓存 搜索推荐
漫画:“排序算法” 大总结
冒泡排序: 漫画:什么是冒泡排序? 选择排序: 漫画:什么是选择排序? 插入排序: 漫画:什么是插入排序? 此外还有冒泡排序的变种,鸡尾酒排序: 漫画:什么是鸡尾酒排序?
167 0
漫画:“排序算法” 大总结
|
算法 搜索推荐
漫画:什么是基数排序?
数组每一个下标位置的值,代表了数列中对应整数出现的次数。 有了这个“统计结果”,排序就很简单了。直接遍历数组,输出数组元素的下标值,元素的值是几,就输出几次: 0,1,1,2,3,3,3,4,4,5,5,6,7,7,8,9,9,9,9,10 显然,这个输出的数列已经是有序的了。 这就是计数排序的朴素版本。
154 0
漫画:什么是基数排序?
|
算法 搜索推荐 Java
漫画:什么是桶排序?
让我们先来回顾一下计数排序: 计数排序需要根据原始数列的取值范围,创建一个统计数组,用来统计原始数列中每一个可能的整数值所出现的次数。
175 0
漫画:什么是桶排序?
漫画:什么是插入排序?
人们如何进行扑克牌的排序呢? 举个例子,比如我手中有红桃6,7,9,10这四张牌,已经处于升序排列:这时候,我又抓到了一张红桃8,如何让手中的五张牌重新变成升序呢?用冒泡排序,选择排序,亦或是快速排序?
169 0
漫画:什么是插入排序?