时间复杂度与空间复杂度的详解

目录

1.时间复杂度

2.时间复杂度计算例题

3.空间复杂度


1.时间复杂度

算法中的基本操作的执行次数,为算法的时间复杂度。
如何表达 时间复杂度?
大O的渐进表示法
实际中我们计算时间复杂度时,我们其实并不一定要计算精确的执行次数,而只需要 大概执行次数,那么这里我们 使用大 O 的渐进表示法。
1、用常数1取代运行时间中的所有加法常数。
2、在修改后的运行次数函数中,只保留最高阶项。
3、如果最高阶项存在且不是1,则去除与这个项目相乘的常数。得到的结果就是大O

举例:

 

// 请计算一下 func1 基本操作执行了多少次?
void func1 ( int N ){
int count = 0 ;
for ( int i = 0 ; i < N ; i ++ ) {
for ( int j = 0 ; j < N ; j ++ ) {
count ++ ;
}
}
for ( int k = 0 ; k < 2 * N ; k ++ ) {
count ++ ;
}
int M = 10 ;
while (( M -- ) > 0 ) {
count ++ ;
}
System . out . println ( count );
}

 题解:

Func1 执行的基本操作次数 :
F(N)=N^2+2*N+10;
(1) 用常数1取代运行时间中的所有加法常数。
F(N)=N^2+2*N+1;
(2) 在修改后的运行次数函数中,只保留最高阶项。
F(N)=N^2;
=>O(N^2);

通过上面我们会发现大O的渐进表示法去掉了那些对结果影响不大的项,简洁明了的表示出了执行次数。  

另外有些算法的时间复杂度存在最好、平均和最坏情况:
最坏情况:任意输入规模的最大运行次数 ( 上界 )
平均情况:任意输入规模的期望运行次数
最好情况:任意输入规模的最小运行次数 ( 下界 )
在实际中一般情况关注的是算法的最坏运行情况,如:数组中搜索数据时间复杂度为 O(N)
O(N)中N表示问题的规模

2.时间复杂度计算例题

例题1:

// 计算 func2 的时间复杂度?
void func2 ( int N , int M ) {
int count = 0 ;
for ( int k = 0 ; k < M ; k ++ ) {
count ++ ;
}
for ( int k = 0 ; k < N ; k ++ ) {
count ++ ;
}
System . out . println ( count );
}

答案及分析:

基本操作执行了M+N次,有两个未知数MN,时间复杂度为 O(N+M) 

例题2:

// 计算 func3 的时间复杂度?
void func3 int N ) {
int count = 0 ;
for ( int k = 0 ; k < 100 ; k ++ ) {
count ++ ;
}
System . out . println ( count );
}

 答案及分析:

基本操作执行了100次,通过推导大O阶方法,时间复杂度为 O(1)

 例题3:

// 计算 bubbleSort 的时间复杂度?
void bubbleSort ( int [] array ) {
for ( int end = array . length ; end > 0 ; end -- ) {
boolean sorted = true ;
for ( int i = 1 ; i < end ; i ++ ) {
if ( array [ i - 1 ] > array [ i ]) {
Swap ( array , i - 1 , i );
sorted = false ;
}
}
if ( sorted == true ) {
break ;
}
}
}

 答案及分析:

O(N)中N表示问题的规模

F(N)=N*(N-1)=N^2-N;

通过推导大 O 阶方法 + 时间复杂度一般看最坏,时间 复杂度为 O(N^2)

 例题4:

// 计算 binarySearch 的时间复杂度?
int binarySearch ( int [] array , int value ) {
int begin = 0 ;
int end = array . length - 1 ;
while ( begin <= end ) {
int mid = begin + (( end - begin ) / 2 );
if ( array [ mid ] < value )
begin = mid + 1 ;
else if ( array [ mid ] > value )
end = mid - 1 ;
else
return mid ;
}
return - 1 ;
}

 答案及分析:

方法1:

对于不能直接看出的并较复杂的问题,可以采用数学归纳法

 答案:

 方法2:

 

N/(2^x) =1(x为循环的执行次数)

x的解:

例题 5

// 计算阶乘递归 factorial 的时间复杂度?
long factorial ( int N ) {
return N < 2 ? N : factorial ( N - 1 ) * N ;
}

对于不能直接看出的并较复杂的问题,可以采用数学归纳法,但对于递归我们有专门总结的方法。

F(N)=递归的次数*每次递归代码的执行次数

 答案及分析:

通过计算分析发现基本操作递归了 N次, 每次递归代码的执行次数为1 时间复杂度为O(N)

例题6:

// 计算斐波那契递归 fifibonacci 的时间复杂度?
int fifibonacci ( int N ) {
return N < 2 ? N : fifibonacci ( N - 1 ) + fifibonacci ( N - 2 );
}

  答案及分析:

对于不能直接看出的并较复杂的问题,可以采用数学归纳法(不展开)

面对这种多递归入口的题,可以使用补全法。

何为补全法?

以F4为例

F(N): 

 

1+2+4+……+2^(N-1)
=2^N-1;
O(2^N)

3.空间复杂度

空间复杂度是对一个算法在运行过程中 临时占用存储空间大小的量度 。空间复杂度不是程序占用了多少 bytes 的空 间,因为这个也没太大意义,所以空间复杂度算的是变量的个数
空间复杂度计算规则基本跟时间复杂度类似 ,也 使用 O 渐进表示法
无论什么类型,只看开了多少的空间

 例题1:

// 计算 bubbleSort 的空间复杂度?
void bubbleSort ( int [] array ) {
for ( int end = array . length ; end > 0 ; end -- ) {
boolean sorted = true ;
for ( int i = 1 ; i < end ; i ++ ) {
if ( array [ i - 1 ] > array [ i ]) {
Swap ( array , i - 1 , i );
sorted = false ;
}
}
if ( sorted == true ) {
break ;
}
}
}

   答案及分析:

 使用了常数个额外空间,所以空间复杂度为 O(1)

例题2:

// 计算 fifibonacci 的空间复杂度?
int [] fifibonacci ( int n ) {
long [] fifibArray = new long [ n + 1 ];
fifibArray [ 0 ] = 0 ;
fifibArray [ 1 ] = 1 ;
for ( int i = 2 ; i <= n ; i ++ ) {
fifibArray [ i ] = fifibArray [ i - 1 ] + fifibArray [ i - 2 ];
}
return fifibArray ;
}

 答案及分析:

动态开辟了N个空间,空间复杂度为 O(N)

例题3:

// 计算阶乘递归 Factorial 的空间复杂度?
long factorial ( int N ) {
return N < 2 ? N : factorial ( N - 1 ) * N ;
}

  答案及分析:

递归调用了 N 次,开辟了 N 个栈帧,每个栈帧使用了常数个空间。空间复杂度为 O(N)

 

以上为我个人的小分享,如有问题,欢迎讨论!!! 

都看到这了,不如关注一下,给个免费的赞 

 

 

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

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

相关文章

nginx基于源码安装的方式对静态页面、虚拟主机(IP、端口、域名)和日志文件进行配置

一.静态页面 1.更改页面内容 2.更改配置文件 3.测试 二.虚拟主机配置 1.基于IP &#xff08;1&#xff09;在html目录下新建目录存放测试文件 &#xff08;2&#xff09;修改nginx.conf文件&#xff0c;在htttp模块中配置两个server模块分别对应两个IP &#xff08;3&am…

外部节点访问 k8s 集群内的 starrocks

问题描述 用kubeadm在虚拟机搭建了k8s&#xff0c;按starrocks官网步骤&#xff0c;用k8s部署了starrocks 部署成功&#xff1a; 在 k8s集群内节点访问到 sr&#xff1a;&#xff08;通过 clusterIP &#xff09; mysql -h 10.97.182.109 -uroot -P 9030 k8s 节点内访问成功…

创建CREATE_STAT_TABLE 统计信息表在达梦和oracle中的使用

达梦 创建CREATE_STAT_TABLE 统计信息表 PROCEDURE CREATE_STAT_TABLE ( STATOWN VARCHAR(128), STATTAB VARCHAR(128), TABLESPACE VARCHAR(128) DEFAULT NULL, GLOBAL_TEMPORARY BOOLEAN DEFAULT FALSE ); 创建普通表的对应系统表的列名字段包括以下&#xff1a; OWNER TABL…

Linux MQTT智能家居项目(智能家居界面布局)

文章目录 前言一、创建工程项目二、界面布局准备工作三、正式界面布局总结 前言 一、创建工程项目 1.选择工程名称和项目保存路径 2.选择QWidget 3.添加保存图片的资源文件&#xff1a; 在工程目录下添加Icon文件夹保存图片&#xff1a; 将文件放入目录中&#xff1a; …

大数据课程I1——Kafka的概述

文章作者邮箱&#xff1a;yugongshiyesina.cn 地址&#xff1a;广东惠州 ▲ 本章节目的 ⚪ 了解Kafka的概念&#xff1b; ⚪ 掌握Kafka的配置与启动&#xff1b; 一、简介 1. 基本概念 Apache kafka 是一个分布式数据流平台。可以从如下几个层面来理解&#x…

ngrok内网穿透可以实现资源共享吗?快解析更加简洁

随着互联网的高速发展&#xff0c;越来越多的人开始意识到内网穿透技术的重要性。在这一技术中&#xff0c;ngrok已经成为了一个备受关注的工具。然而&#xff0c;很多人对于ngrok是否可以进行资源共享存在疑问。本文将从新的角度出发&#xff0c;深入探讨这个问题。 了解什么…

TEC2083BS-PD码转换器(解决博世矩阵控制PELCO派尔高球机的问题)

TEC2083BS-PD码转换器 使用说明 1.设备概述 控制码转换器在安防工程中起着非常重要的角色&#xff0c;随着高速球型摄像机在安防工程中大范围的使用&#xff0c;而高速球厂家都因为某些原因很少使用博世、飞利浦的协议。为此&#xff0c;工程商经常会遇到博世协议和PELCO协议之…

【Linux】云服务器自动化部署VuePress博客(Jenkins)

前言 博主此前是将博客部署在 Github Pages&#xff08;基于 Github Action&#xff09;和 Vercel 上的&#xff0c;但是这两种部署方式对于国内用户很不友好&#xff0c;访问速度堪忧。因此将博客迁移到自己的云服务器上&#xff0c;并且基于 Jenkins&#xff08;一款开源持续…

eNSP:双向重定向和路由策略练习

实验要求&#xff1a; 拓扑图&#xff1a; IP、路由器 r1: <Huawei>sys [Huawei]sys r1 [r1]int g 0/0/0 [r1-GigabitEthernet0/0/0]ip add 12.1.1.1 24 [r1-GigabitEthernet0/0/0]int g 0/0/1 [r1-GigabitEthernet0/0/1]ip add 14.1.1.1 24 [r1-GigabitEthernet0/0/1]…

腾讯云轻量服务器和云服务器的CPU处理器有差别吗?

腾讯云轻量应用服务器和CVM云服务器的CPU处理器性能有差别吗&#xff1f;创建轻量应用服务器时不支持指定底层物理服务器的CPU型号&#xff0c;腾讯云将随机分配满足套餐规格的物理CPU型号&#xff0c;通常优先选择较新代次的CPU型号。而云服务器CVM的CPU处理器型号、主频都是有…

flutter开发实战-TextPainter计算文本内容的宽度

flutter开发实战-TextPainter计算文本内容的宽度 最近开发过程中根据Text文本的大小判断是否需要进行显示跑马灯效果&#xff0c;获取文本的大小&#xff0c;需要TextPainter来获取Size 一、TextPainter TextPainter主要用于实现文本的绘制。TextPainter类可以将TextSpan渲染…

docker下载和案例

文章目录 Docker安装一,根据官方文档安装二,根据我以下方式 Docker配置错误导致漏洞一,CRLF注入漏洞介绍在nginx中该漏洞例子解决方法 目录穿越漏洞介绍解决方法 Docker安装 一,根据官方文档安装 官方文档 二,根据我以下方式 docker安装要求&#xff1a; Docker要求Ce…

Java List(列表)

List 是一个有序、可重复的集合&#xff0c;集合中每个元素都有其对应的顺序索引。List 集合允许使用重复元素&#xff0c;可以通过索引来访问指定位置的集合元素。List 集合默认按元素的添加顺序设置元素的索引&#xff0c;第一个添加到 List 集合中的元素的索引为 0&#xff…

pytest接口自动化测试框架搭建的全过程

目录 一. 背景 二. 基础环境 三. 项目结构 四、框架解析 pytest是Python的一种单元测试框架,可用来组织用例执行,用例断言,下面这篇文章主要给大家介绍了关于pytest接口自动化测试框架搭建的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下 一. 背景 Pyte…

HTTP协议

HTTP协议 应用层再谈 "协议"网络版计算器 HTTP协议认识URLurlencode和urldecodeHTTP协议格式HTTP的方法HTTP的状态码HTTP常见Header HTTPS协议HTTPS 是什么什么是"加密"为什么要加密常⻅的加密⽅式 HTTPS 的⼯作过程探究⽅案 1 - 只使⽤对称加密⽅案 2 - 只…

【Linux】TCP协议的相关实验——深入理解

TCP相关实验 理解CLOSE_WAIT状态 当客户端和服务器在进行TCP通信时&#xff0c;如果客户端调用close函数关闭对应的文件描述符&#xff0c;此时客户端底层操作系统就会向服务器发起FIN请求&#xff0c;服务器收到该请求后会对其进行ACK响应。 但如果当服务器收到客户端的FIN…

Apikit 自学日记:API 异常监控-监控报告

在 api 管理中&#xff0c;查看 api 异常监控的监控报告&#xff0c;在 apikit 中也是常用的功能&#xff0c;通常你可以在流程综合报告页中看到当前流程在选定时间段内的整体监控情况... 在 APIkit 中监控报告有这几种类别&#xff1a; 单接口监控报告 流程监控报告 项目监控…

linux静态库与动态库

1、动态库和静态库概念 Linux中的库分为动态库和静态库。 静态库&#xff08;.a&#xff09;&#xff1a;库文件以.a为后缀&#xff0c;程序在编译链接时把库的代码链接到可执行文件中&#xff08;将需要的库函数拷贝一份到代码中&#xff09;。程序运行时不需要再跳转到静态…

layui 集成 ztree异步加载

首先&#xff0c;layui环境搭建&#xff0c;ztree环境引入 ztree的js和css都要引入&#xff0c;我这里暂时用的是core包> 静态&#xff0c;一句话就够了 <!-- 左侧菜单树形组件 --><div class"layui-col-md3"><div class"layui-footer "…

计算机网络(7) --- UDP协议和TCP协议

计算机网络&#xff08;6&#xff09; --- https协议_哈里沃克的博客-CSDN博客https协议https://blog.csdn.net/m0_63488627/article/details/132112683?spm1001.2014.3001.5501 目录 1.补充知识 1.PORT端口号 2.端口号范围划分 3.知名端口号 2.UDP协议 1.UDP报头 2.U…