漫画:什么是时间复杂度

转载自    漫画:什么是时间复杂度

时间复杂度的意义

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

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

 

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

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

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

于是......

 

 

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

1.运行时间

2.占用空间

 

 

基本操作执行次数

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

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

 

答案自然是 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天吃掉一个鸡腿。那么小灰吃掉整个鸡腿需要多少天呢?

答案自然是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

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

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

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

vo

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

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

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

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

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

void eat4(int n){for(int i=0; i<n; i++){for(int j=0; j<i; j++){System.out.println("等待一天");}System.out.println("吃一寸面包");}
}

渐进时间复杂度

有了基本操作执行次数的函数 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表示法

 

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

  1. 如果运行时间是常数量级,用常数1表示。

  2. 只保留时间函数中的最高阶项

  3. 如果最高阶项存在,则省去最高阶项前面的系数。

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

场景1:

T(n) = 3n 

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

T(n) =  O(n)

 

 

场景2:

T(n) = 5logn 

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

T(n) =  O(logn)

 

 

场景3:

T(n) = 2

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

T(n) =  O(1)

 

场景4:

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

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

T(n) =  O(n^2)

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

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

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

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

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

 

时间复杂度的巨大差异

 

 

我们来举过一个栗子:

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

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

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

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

 

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

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

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/325977.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Office基础和计算机操作基础的知识点(一)

 运指如飞&#xff1a; 一、鼠标的基本操作 1.单击 2.双击 3.移动 4.拖拽 二、键盘的分布 1.功能键区 2.打字键盘区 3.数字键盘区 4.指示灯 5.编辑键区 三、输入法 1.切换输入法&#xff1a;ctrlshift 2.选中的方法&#xff1a; 1)鼠标选中 2)按shift键移动左右键 四、记事…

Flume整合SparkStreaming出现UnsupportedClassVersionError: org/apache/spark/streaming/flume/sink/SparkFlum

Flume整合SparkStreaming使用Poll方式拉取数据出现 UnsupportedClassVersionError: org/apache/spark/streaming/flume/sink/SparkFlumeProtocol : Unsupported major.minor version 52.0 问题&#xff1a; INFO node.Application: Starting Source source1 18/06/12 03:34…

Visual Studio 和 Team Foundation Server 产品维护及周期策略

1 适用于 Visual Studio 和 Team Foundation Server 2012 - 2017 这些产品遵循 Microsoft 10 年支持生命周期策略&#xff08;5 年主流支持和 5 年扩展支持&#xff09;&#xff0c;起始日期为主要产品版本向全球 (RTW) 发布的日期。 例如&#xff0c;Visual Studio 2017 于 20…

28. 实现 strStr()---LeetCode---JAVA(今天又是一行超人)

class Solution {public int strStr(String haystack, String needle) {return haystack.indexOf(needle);} }

DevOps通用及版本控制面试题

转载自 DevOps通用及版本控制面试题 通用DevOps面试问题 此类别将包含与任何特定DevOps阶段无关的问题。这里的问题旨在测试您对DevOps的理解&#xff0c;而不是关注特定工具或阶段。 问题一&#xff1a; DevOps和Agile之间的根本区别是什么&#xff1f; 两者之间的差异…

Office基础知识点总结(二)

 PowerPoint幻灯片制作&#xff1a; 一、PPT2010的界面&#xff1a; 1.功能选项卡 2.大纲区 3.标题栏 4.备注页 5.状态栏 6.工作区 二、创建PPT演示文稿 1.开始–》PPT2010 2.在PPT里面新建–》文件–》新建–》样本模板–》选择模板。 三、PPT2010中的视图&#xff1a; 1.普…

zookeeper出现Error contacting service. It is probably not running.

在两个节点 的zookeeper搭建启动的时候出现如下报错 JMX enabled by default Using config: /root/app/zookeeper/bin/../conf/zoo.cfg Error contacting service. It is probably not running. 查看zookeeper.out 日志报错如下 2018-06-13 03:13:18,573 [myid:0] - INFO …

微信和支付宝支付模式详解及实现(.Net标准库)

支付基本上是很多产品都必须的一个模块&#xff0c;大家最熟悉的应该就是微信和支付宝支付了&#xff0c;不过更多的可能还是停留在直接sdk的调用上&#xff0c;甚至和业务系统高度耦合&#xff0c;网上也存在各种解决方案&#xff0c;但大多形式各异&#xff0c;东拼西凑而成。…

DFS

//深度优先算法//i第一次就是0public void dfs(boolean[]isVisited,int i){//首先我们访问该节点System.out.println(getValueByIndex(i));isVisited[i]true;int wgetFirstNeighbor(i);while (w!-1){if(!isVisited[w]){dfs(isVisited,w);}//如果w节点已经被访问过wgetNextNeigh…

这些Spring面试题,你都会了吗?

转载自 这些Spring面试题&#xff0c;你都会了吗&#xff1f; 问题一&#xff1a; 什么是Spring&#xff1f; Spring是Enterprise Java的开源开发框架。Spring Framework的核心功能可用于开发任何Java应用程序&#xff0c;但有一些扩展用于在Java EE平台之上构建Web应用程…

计算机操作基础知识点总结

 计算机组装&#xff1a; 一、计算机硬件组成&#xff1a; 1.CPU 2.主板 3.内存 4.硬盘 5.光驱 6.显卡 7.网卡 8.声卡 9.电源 10.操作系统 二、CPU型号&#xff1a; 1.INTEL 1)主频&#xff1a;2.8-3.7GHZ 2)核心&#xff1a;双核、四核、八核、十核 3)高速总缓冲&#xff1a…

Kafka启动出现Error: Could not create the Java Virtual Machine. Error: A fatal exception has occurred. Pr

kafka启动时&#xff0c;出现报错&#xff01; [rootSpark1 bin]# cat nohup.out Unrecognized VM option ‘UseCompressedOops’ Error: Could not create the Java Virtual Machine. Error: A fatal exception has occurred. Program will exit. 原因&#xff1a; 开始…

BFS

public void bfs(boolean[]isVisited,int i){int u;//表示队列的头结点对应的下标int w;//表示邻接节点w//队列LinkedList queue new LinkedList();//访问节点&#xff0c;输出节点信息System.out.println(getValueByIndex(i)">");isVisited[i]true;queue.addLas…

.NET遇上Docker - 使用Docker Compose组织Ngnix和.NETCore运行

本文工具准备&#xff1a; Docker for WindowsVisual Studio 2015 与 Visual Studio Tools for Docker或 Visual Studio 2017 需要在安装时选择“容器开发支持”&#xff0c;如图&#xff1a; Docker的思想是将不同的应用放在不同的容器中分开运行&#xff0c;如运行.NetCore …

平面设计基础(PS)知识点总结

 PS基础知识&#xff1a; 一、ps全称&#xff1a;Adobe Photoshop &#xff0c;Adobe Photoshop是由Adobe Systems 开发和发行的专门用于图形图像处理的软件。 二、PS的应用领域&#xff1a; 1.在平面设计中的应用 2.在插画设计中的应用 3.在网页设计中的应用 4.在界面设计中…

为什么大公司一定要使用DevOps

转载自 为什么大公司一定要使用DevOps 0 DevOps的意图 究竟什么是DevOps? 要想回答这个问题&#xff0c;首先要明确DevOps这个过程参与的人员是谁&#xff1f;即开发团队和IT运维团队&#xff01;那么&#xff0c;DevOps的意图是什么呢&#xff1f;即在两个团队之间&#…

基于ZooKeeper实现HA高可用性以及自动主备切换

默认情况下&#xff0c;standalone cluster manager对于worker节点的失败是具有容错性的&#xff08;迄今为止&#xff0c;Spark自身而言对于丢失部分计算工作是有容错性的&#xff0c;它会将丢失的计算工作迁移到其他worker节点上执行&#xff09;。然而&#xff0c;调度器是依…

怎样访问Springboot项目中static中的资源

第一步 成功启动 第二步 输入 协议://ip:端口/资源的名字 eg http://localhost:8080/06-demo-point.html

数据库基础的知识点

 数据库系统基础&#xff1a; 一、数据库&#xff1a;存储数据的仓库 二、数据库能够做什么&#xff1a; 1.存储大量的数据&#xff0c;方便检索和访问。 2.保持数据的信息一致、完整 3.共享和安全 4.通过组合分析&#xff0c;产生新的有用信息。 三、应用程序和数据库的关系…

asp.net core源码飘香:Logging组件

简介&#xff1a; 作为基础组件&#xff0c;日志组件被其他组件和中间件所使用&#xff0c;它提供了一个统一的编程模型&#xff0c;即不需要知道日志最终记录到哪里去&#xff0c;只需要调用它即可。 使用方法很简单&#xff0c;通过依赖注入ILogFactory&#xff08;CreateL…