【从0到1学算法】大O表示法
一般我们在选择算法时,都是想要选择效率最高的算法。那算法的效率,用什么表示?没错!就是用大O表示法。
PS: 大O表示法中,log即为log2,后面不再说明。
下面以简单查找和二分查找,在含有n个元素的有序列表中查找其中一个元素为例,下表总结了我们发现的情况。
使用简单查找时,最多需要猜测次数与列表长度相同,这被称为线性时间,大O表示法为O(n)。
二分查找则不同,最多需要猜测次数为logn(n为列表长度),这被称为对数时间(log时间),大O表示法为O(logn)。
基本概念
大O表示法指出了算法的速度有多快。
可能你会好奇,它的单位是多少?秒?没有单位,它并非指的是时间,而是从增量的角度衡量。
列表中查找元素,简单查找、二分查找的增速如下图。
假若我们不知道增速,只知道查找100个元素时的查找时间,猜测10000个元素时的查找时间:
对于简单查找,100个元素时为100毫秒,简单推算出10000个元素为10秒;
对于二分查找,100个元素时为7毫秒,简单推算出10000个元素为700毫秒。
PS:简单推算 10000个元素时的运行时间= 运行时间(100个元素时)* 100
简单查找的推算是对的,因为的增速是n,而二分查找的推算是错的,它的增速为logn,这便不能理所当然简单推算了。
很显然,我们只要知道算法的增速,便能知道它在n个元素中运行的运行时间了,大O表示法就是用来表示算法增速的。
专业描述:大O表示法表示操作数的增速,指出了算法运行时间的增速。
常见的大O运行时间(从快到慢)
- O(㏒n) 对数时间 比如二分查找
- O(n) 线性时间 比如简单查找
- O(n*㏒n) 比如快速排序
- O(n²) 比如选择排序
- O(n!) 比如旅行者问题
大O表示法的不同维度
时间复杂度
上述的大O表示法都是用来表示时间复杂度,而且通常指的是最坏情况下的时间复杂度。
通常有以下3种时间复杂度:
以简单查找为例子,在n个元素的列表中查找目标元素
- 最好情况时间复杂度:目标元素刚好在列表第一个位置,那么只需要一次就能找到,时间复杂度为O(1)。
- 最坏情况时间复杂度:目标元素在列表最后一个位置或者不在列表中,那么得需要遍历完整个列表才能得出结果,时间复杂度为O(n)。
- 平均情况时间复杂度:考虑目标元素在列表中任何位置的情况。 下面简单分析下:目标元素如果在列表中,出现的位置有n种情况,加上不在列表中这一种情况,总共n+1种情况。每种情况下需要遍历的次数如下表: 目标元素所在位置 遍历次数 第1个位置 1 第2个位置 2第3个位置 3第n个位置 n 不在列表中 n 平均遍历次数=各种情况遍历次数相加÷总的情况数 各种情况遍历次数相加为((1+2+3...+n)+n),总的情况数(n+1)
平均情况复杂度为:
((1+2+3...+n)+n)/(n+1)=n(n+3)2(n+1)
大O表示法,会省略系数、低阶、常量,所以平均情况时间复杂度是O(n)。
空间复杂度
空间复杂度是对一个算法在运行过程中临时占用存储空间大小的一个量度,反映的是一个趋势。
空间复杂度比较常用的有:O(1)、O(n)、O(n²),我们下面来看看:
空间复杂度 O(1)
如果算法执行所需要的临时空间不随着某个变量n的大小而变化,即此算法空间复杂度为一个常量,可表示为 O(1)。
例子:
int i = 1;
int j = 2;
++i;
j++;
int m = i + j;
i、j、m 所分配的空间都不随着处理数据量变化,因此它的空间复杂度 O(1)。
空间复杂度 O(n)
int[] m = new int[n]
for(i=1; i<=n; ++i)
{
j = i;
j++;
}
第一行new了一个长度为n的数组m,占用大小为n,后面虽然有循环,但没有再分配新的空间,因此,这段代码的空间复杂度主要看第一行即可,即O(n)。
PS:O(n²)的异同,不再累述。
以下常见排序算法的复杂度分析,感受一下时间复杂度和空间复杂度。
参考:《算法图解》 https://blog.csdn.net/jsjwk/article/details/84315770
https://www.cnblogs.com/jonins/p/9956752.html
- ROWNUMBER() OVER( PARTITION BY COL1 ORDER BY COL2)/ ROWNUMBER() OVER( PARTITION BY COL1 ORDER BY CO
- 干货 | 前端常用的通信技术
- TP-LINK WR941N路由器研究
- ORA-01113问题的简单分析(r6笔记第3天)
- Tensorflow 中 learning rate decay 的奇技淫巧
- hive数据:名词解释
- 巧妙使用exchange partition的一个案例(r6笔记第1天)
- r与rjava
- 使用expect运行动态脚本(r6笔记第19天)
- 数据库日志中一条"异常"信息所包含的细节(r6笔记第18天)
- linux下编辑VI窗口插入与编辑命令
- rman中三个不完全恢复场景(r6笔记第16天)
- 关于delete,drop,truncate的问题 (r6笔记第14天)
- R语言进行分析,比较详细的一篇,亲测过哦
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法