复杂度分析之大O表示法。

2024-02-08 02:18

本文主要是介绍复杂度分析之大O表示法。,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、概要。
好的程序设计表现出程序执行速度快,占用内存空间小的优点,分别对应时间复杂度和空间复杂度。
2、复杂度分类两种。
算法的时间复杂度是指算法需要消耗的时间资源。算法执行时间的增长率与f(n) 的增长率正相关,称作渐近时间复杂度(Asymptotic Time Complexity),简称时间复杂度。定义为T(n) = O(f(n)),称函数T(n)以f(n)为界或者称T(n)受限于f(n)。
算法的空间复杂度是指算法需要消耗的空间资源。其计算和表示方法与时间复杂度类似,一般都用复杂度的渐近性来表示。

常见的算法时间复杂度由小到大依次为:
O(1) 常数阶
O(logn) 对数阶
O(n) 线性阶
O(n*logn) 线性对数阶
O(n^2) 平方阶
O(n^k) K方阶
O(2^n) 指数阶
O(n!) 阶乘阶

其中,O(2^n) 和O(n!)为非多项式量级,NP(Non-Deterministic Polynomial,非确定多项式)问题,其复杂度计算机往往不能承受的。除此二者,都是多项式量级。
在这里插入图片描述
图表来源:https://www.bigocheatsheet.com

3、时间复杂度。
3.1辨识要素及示例。

int CalcSum(int n)
{int sum = 0;for (int i = 0; i < n; i++){sum += i;}return sum;
}

Demo求累加和,假设每条语句执行时间为time,容易算出来该函数运行时间为:T(n) = time+time+ntime2 = 2*(n+1)*time。用f(n)来表示代码的执行次数和数据规模的关系,即f(n)=2n+2。f(n)中的常数项对于整个公式的值的影响我们直接忽略,同样,我们也忽略系数。即得:f(n) = n;代入T(n) = O(f(n)),得:T(n) = O(n),线性阶。

①、只关注循环执行次数最多的一段代码;


int CalcSum(int n)
{int sum1 = 0;int sum2 = 0;for (int i = 0; i < n; i++){sum1 += i;}for (int j = 0; j < n * 2; j++){sum2 += j;}return sum1 + sum2;
}

容易看出来循环二比循环一的执行次数多,因此只需要关注循环二,得到T(n) = O(n);线性阶。
②、加法法则:总复杂度等于量级最大的那段代码的复杂度;

int CalcSum(int n)
{int sum1 = 0;int sum2 = 0;int sum3 = 0;for (int i = 0; i < 100; i++){sum1 += i;}for (int j = 0; j < n; j++){sum2 += j;}for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){sum3 += j;}}return sum1 + sum2 + sum3;
}

容易得到循环一二三的时间复杂度分别为:O(1) (常数阶),O(n)(线性阶),O(n^2)(平方阶);则T(n) = O(1)+O(n)+O(n^2);按照加法原则,总复杂度等于量级最大的那段代码的复杂度。即T(n) = O(n^2),平方阶。
③、乘法法则:嵌套代码的复杂度等于嵌套内外代码复杂度的乘积。


int CalcSum(int n)
{int sum = 0;for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){sum += j;}}return sum;
}

嵌套循环 = 外层内层。即T(n) = O(nn) = O(n^2),平方阶。
Ps:
对数阶:

int CalcSum(int n)
{int sum = 0;int i = 1;do{i = i * 2;sum += i;} while (i < n);return sum;
}

在while循环里面,每次都将 i 乘以 2翻倍,距离n越来越近。循环x次后,退出,也就是说2^x >=n。那么x = log2n(以2为底,n的对数)。这就是对数阶,时间复杂度为:T(n) = O(logn)。
线性对数阶:
将对数阶循环n遍,就是线性对数阶。即:T(n) = n*(O(logn)) = O(n*logn)。
3.2时间复杂度三种类别。
时间复杂度里细分起来又有最好、最坏、平均情况时间复杂度之分:
1、最好情况时间复杂度就是在最理想的情况下,执行这段代码的时间复杂度;
2、最坏情况时间复杂度就是在最糟糕的情况下,执行这段代码的时间复杂度;
3、平均情况时间复杂度顾名思义就是结合概率论分析从最好到最坏每种情况平均下来的加权平均时间复杂度。
在这里插入图片描述
图表来源:https://www.bigocheatsheet.com
4、空间复杂度。
空间复杂度比较常用的有:O(1)、O(n)、O(n²)。
4.1 S(n) = O(1).
不存在存储空间随变量变化情况。

void GetSpace(int n)
{vector<int> array;array.push_back(1);array.push_back(2);array.push_back(3);array.push_back(4);array.push_back(5);int sum = 0;for (auto it:array){sum += it;}
}

4.2 S(n) = O(n).
例程中,申请空间后,没有在申请其他空间。复杂度为:O(n)。

void GetSpace(int n)
{int* p = new int[n];int sum = 0;for (int i = 0; i <= n; ++i){sum += i;}
}

5、 总结
①、算法的速度并非指时间,不是以秒为单位;而是操作数的增速。从增量的角度度量的。
②、平时说算法的速度,指的是随着输入的增加,其运行时间将会以什么样的速度进行增加。
③、算法运行时间用大O表示法表示。
④、 O(㏒n)比O(n)快。当操作的元素规模差距越大,快的越明显。

这篇关于复杂度分析之大O表示法。的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/689640

相关文章

Spring Boot Interceptor的原理、配置、顺序控制及与Filter的关键区别对比分析

《SpringBootInterceptor的原理、配置、顺序控制及与Filter的关键区别对比分析》本文主要介绍了SpringBoot中的拦截器(Interceptor)及其与过滤器(Filt... 目录前言一、核心功能二、拦截器的实现2.1 定义自定义拦截器2.2 注册拦截器三、多拦截器的执行顺序四、过

C++ scoped_ptr 和 unique_ptr对比分析

《C++scoped_ptr和unique_ptr对比分析》本文介绍了C++中的`scoped_ptr`和`unique_ptr`,详细比较了它们的特性、使用场景以及现代C++推荐的使用`uni... 目录1. scoped_ptr基本特性主要特点2. unique_ptr基本用法3. 主要区别对比4. u

Nginx内置变量应用场景分析

《Nginx内置变量应用场景分析》Nginx内置变量速查表,涵盖请求URI、客户端信息、服务器信息、文件路径、响应与性能等类别,这篇文章给大家介绍Nginx内置变量应用场景分析,感兴趣的朋友跟随小编一... 目录1. Nginx 内置变量速查表2. 核心变量详解与应用场景3. 实际应用举例4. 注意事项Ng

Java多种文件复制方式以及效率对比分析

《Java多种文件复制方式以及效率对比分析》本文总结了Java复制文件的多种方式,包括传统的字节流、字符流、NIO系列、第三方包中的FileUtils等,并提供了不同方式的效率比较,同时,还介绍了遍历... 目录1 背景2 概述3 遍历3.1listFiles()3.2list()3.3org.codeha

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe