复杂度分析之大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

相关文章

MySQL设置密码复杂度策略的完整步骤(附代码示例)

《MySQL设置密码复杂度策略的完整步骤(附代码示例)》MySQL密码策略还可能包括密码复杂度的检查,如是否要求密码包含大写字母、小写字母、数字和特殊字符等,:本文主要介绍MySQL设置密码复杂度... 目录前言1. 使用 validate_password 插件1.1 启用 validate_passwo

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

Olingo分析和实践之EDM 辅助序列化器详解(最佳实践)

《Olingo分析和实践之EDM辅助序列化器详解(最佳实践)》EDM辅助序列化器是ApacheOlingoOData框架中无需完整EDM模型的智能序列化工具,通过运行时类型推断实现灵活数据转换,适用... 目录概念与定义什么是 EDM 辅助序列化器?核心概念设计目标核心特点1. EDM 信息可选2. 智能类