Python 数据结构 2.时间复杂度和空间复杂度

Life is a journey

                          —— 25.2.28

一、引例:穷举法

1.单层循环

所谓穷举法,就是我们通常所说的枚举,就是把所有情况都遍历了的意思。

例:给定n(n ≤ 1000)个元素ai,求其中奇数有多少个

判断一个数是偶数还是奇数,只需要求它除上2的余数是0还是1,那么我们把所有数都判断一遍,并且对符合条件的情况进行计数,最后返回这个计数器就是答案,这里需要遍历所有的数,这就是穷举

def JudgeNum(self, n: int, a: List[int]) -> int:count = 0for i in range(n):if a[i] % 2 == 1:count += 1return count

时间复杂度 O(n)


2.双层循环

例2:给定n(n ≤ 1000)个元素ai,求有多少个二元组(i,j),满足ai + aj是奇数(i < j)。

def JudgeNum(self, n: int, a[]: List[int]) -> int: count = 0for i in range(n):for j in range(i + 1, n):if (a[i} + a[j]) % 2 == 1:count += 1return count

时间复杂度 O(n^2)


3.三层循环

例3:给定n(n ≤ 1000)个元素ai,求有多少个三元组(i,j,k),满足ai + aj + ak是奇数(i < j < k)。

def JudgeNum(self, n: int, a: list[int]) -> int:count = 0for i in range(n):for j in range(i + 1, n):for k in range(j + 1, n):if (a[i] + a[j] + a[k]) % 2 == 1:count += 1return count 

时间复杂度 O(n^3)

随着循环嵌套的越多,时间消耗会越来越多,并且三个循环是乘法的关系,也就是遍历次数随着n的增加,呈现立方式的增长


4.递归枚举

例:给定n(n ≤ 1000)个元素ai 和 一个整数 k(k ≤ n),求有多少个有序k数组,满足他们的和是偶数

我们需要根据k的不同,决定写几层循环,k的最大值为1000,也就意味着我们要写1000个if-else语句,显然,这样是无法接受的,比较暴力的做法是采用递归


二、时间复杂度

1.时间复杂度的表示法

在进行算法分析时,语句总的执行次数 T(n) 是关于问题规模 n 的函数,进而分析 T(n) 随着 n 的变化情况而确定 T(n) 的数量级

算法的时间复杂度,就是算法的时间度量,记作:T(n)=O(f(n)) 用大写的 O 来体现算法时间复杂度的记法,我们称之为:大 O 记法

Ⅰ、时间函数

时间复杂度往往会联系到一个函数,自变量:表示规模,应变量:表示执行时间。

这里所说的执行时间,是指广义的时间,也就是单位并不是"秒"、"毫秒"这些时间单位,它代表的是一个"执行次数"的概念。我们用  f(n) 来表示这个时间函数。

Ⅱ、经典函数举例

在例1中,我们接触到了单层循环,这里的 n 是一个变量,随着 n 的增大,执行次数增大,执行时间就会增加,所以就有了时间函数的表示法如下:f(n) = n

这就是经典的线性时间函数

在例2中,我们接触到了双层循环,它的时间函数表示法如下:f(n) = n * (n - 1) / 2

这是一个平方级别的时间函数

在例3中,我们接触到了三层循环,它的时间函数表示法如下:f(n) = n * (n - 1) * (n - 2) / 6

这是一个立方级别的时间函数


2.时间复杂度

一个算法中的语句执行次数称为语句频度或时间频度。记为T(n)。

并且我们有一个更加优雅的表示法,即:T(n)=O(f(n)),其中 O 念成 大O

1) 当 f(n) = n,我们称这个算法拥有线性时间复杂度,记作 O(n);

2) 当 f(n) = n * (n-1) / 2,我们称这个算法拥有平方级时间复杂度,记作 O(n^2);

3) 当 f(n) = n * (n-1) * (n-2) / 6,我们称这个算法拥有立方级的时间复杂度,记作 O(n^3);

这时候我们发现,f 的函数可能很复杂,但是 O 表示的函数往往比较简单,它舍弃了一些"细节“


3.高阶无穷小

如果 lim(β / α) = 0,则称 ”β 是比 α较高阶的无穷小“

例:f(n) = n * (n - 1) / 2

共两部分组成,一部分是 n ^ 2 的部分,另一部分是 n 的部分,显而易见,一定是 n ^ 2,相对于 n ^ 2 来说,n 可以忽略不计

随着 n 的增长,线性的部分增长已经跟不上平方部分,这样,线性部分的时间消耗相对于平方部分来说已经”微不足道“,所以我们直接忽略,于是就有时间复杂度表示如下:

T(n) = O(f(n))

        = O(1/2 * n ^ 2 - 1 / 2 * n)

        = O(1/2 * n ^ 2)

        = O(n ^ 2)

所以,它的时间复杂度就是O(n ^ 2)了


4.简化系数

发现上述的公式推到的过程中,将 n ^ 2 前面的系数 1/2 去掉了,这是由于 时间复杂度描述的更多的是一个数量级,所以尽量减少干扰项,对于两个不同的问题,可能执行时间不同,但是我们可以说他们的 时间复杂度 是一样的


三、常见的时间复杂度

1.常数阶

max = 1024
def getMax() -> int:return max  

没有循环,是常数时间,表示为O(1)


2.对数阶

例4:给定n(n ≤ 10000)个元素的有序数组 ai 和 整数 v,求 v 在数组中的下标,不存在则输出 -1

这个问题是一个常见的查询问题,我们可以用O(n)的算法遍历整个数组,然后去找 v 的值

当然,也有更快的方法,注意到题目中的条件,数组 ai 是有序的,所以我们可以利用二分查找来实现

def bin(n: int, a: List[int], v:int) -> int:left = 0,right = n - 1mid = 0while left <= right:mid = (left + right) // 2if a[mid] < v:left = v + 1elif a[mid] > v:right = v - 1else:return midreturn -1

这是一个二分查找的实现,时间复杂度为O(logn)

每次相当于把n切半,即:

n -> n / 2 -> n / 4 -> … -> n / 2 ^ k -> … -> 0

f(n) = O(n) = O(k) = O(logn)


3.根号阶

例5:给定一个数 n(n ≤ 10 ^ 9),问 n 是否是一个素数(素数的概念:除了1和它本身,没有其他因子)

基于素数的概念,我们可以枚举所有 i 属于[2, n),看能否整除 n,一旦能整除,代表找到了一个银子,则不是素数,当所有数枚举完还没找到,他就是素数

但是这样做,显然效率太低,我们进行一些思考,得到如下算法:

import mathdef isPrime(n: int) -> bool:if n <= 1:return Falsesqrtn = math.sqrt(n)for i in range(2, sqrtn + 1):if n % i == 0:return Falsereturn True

这个算法的时间复杂度为:O(根号n)

为什么只需要枚举 根号n 以内的数呢?

因为一旦有一个因子 s,必然有另一个因子 n / s,它们之间必然有一个大小关系,无论是 s ≤ n / s,还是 n / s ≤ s,都能通过两边乘上 s 得出:

比根号n小的数中,如果没有这样一个因子,则比根号n大的数中也不会存在这样一个因子


4.线性阶

例1中,我们接触到了单层循环,这里的 n 是一个变量,随着 n 的增大,执行次数增大,执行时间就会增加,所以就有了时间函数的表示法如下:f(n) = n


5.线性对数阶

例6:给定n(n ≤ 100000)个元素 ai,求满足 ai + aj = 1024 的有序二元组(i, j)有多少对

我们可以先对所有元素 ai 按照递增排序,然后枚举 ai,并且在[i + 1, n)范围内查找是否存在 ai + aj = 1024

def Find(n: int, a: List[int]) -> int:count = 0sort(a)for i in range(n):j = 1024 - a[i]left, right = i + 1, n - 1while left <= right:mid = left + (right - left) // 2if a[mid] == target:count += 1breakelif a[mid] < taegrt:left = mid + 1else:right = mid - 1return count

 f(n) = O(n * logn) = O(nlogn)


6.多项式阶

多项式的含义是函数 f(n) 可以表示成如下形式:

所以 O(n^5)、O(n^4)、O(n^3)、O(n^2)、O(n)都是多项式时间


7.指数阶 

例7:给出n(n ≤ 15)个点,以及每两个点之间的关系(连通还是不连通),求一个最大的集合,使得在这个集合中都连通

这是求子集的问题,由于最多只有 15 个点,我们就可以枚举每个点选或者不选,总共 2^n 种情况,然后再判断是否满足题目中的连通性,这个算法时间复杂度为 O(n ^ 2 * 2 ^ n)


8.阶乘阶

例8:给定n(n ≤ 12)个点,并且给出任意两点间的距离,求从 s 点开始经过所有点回到 s 点的距离的最小值

这个问题就是典型的暴力枚举所有情况求解,可以把这些点当成是一个排列,所以排列方案数为: n!


四、如何判断时间复杂度

1.标准

首先,我们需要一个标准,也就是总的执行次数多少合适

我们把其定义为 S = 10 ^ 6

2.问题规模

有了标准以后,我们还需要知道问题规模,也就是O(n)中的n

3.套公式

然后就是凭感觉套公式了

        当 n < 12 时,可能是需要用到阶乘级别的算法,即 O(n!)

        当 n < 16 时,可能是需要状态压缩的算法,比如 O(n ^ 2)、O(n * 2 ^ n)、O(n ^ 2 * 2 ^ n)

        当 n < 30 时,可能是需要 O(n ^ 4)的算法,因为 30 ^ 4 差不多接近 10 ^ 6

        当 n < 100 时,可能是需要 O(n)的算法,因为 1003= 106

        当n < 1000 时,可能是需要 O(n ^ 2)的算法,因为 1000 ^ 2 = 10 ^ 6

        当n < 100000 时,可能是需要 O(n * log2n)、O(n * (log2n) ^ 2)的算法

        当n < 1000000 时,可能是需要 O(根号n)、O(n)的算法


五、空间复杂度

        空间复杂度是指算法在执行过程中所需的额外存储空间。这包括算法在运行时使用的变量、数组、链表 等数据结构所占用的内存空间。它和算法的时间复杂度一起,是衡量算法性能的重要指标之一。

        在算法设计中,我们通常希望尽可能地降低空间复杂度,以减少内存的使用,提高算法的效率。然而,在某些情况下,为了实现算法的功能,可能需要使用更多的存储空间。


六、常见数据结构的空间复杂度

1.顺序表:O(n),其中 n 是顺序表的长度

2.链表:O(n),其中 n 是链表的长度

3.栈:O(n),其中 n 是 栈的最大深度

4.队列:其中 n 是队列的最大长度

5.哈希表:O(n),其中 n 是哈希表中元素的数量

6.树:O(n),其中 n 是树的节点数量

7.图:O(n + m),其中 n 是图中顶点的数量,m 是图中边的数量


七、空间换时间

通常使用额外空间的目的,就是为了换取时间上的效率,也就是我们常说的 空间换时间

最经典的空间换时间就是动态规划,例如求一个斐波那契数列的第 n 项的值,如果不做任何优化就是利用循环进行计算,时间复杂度 (n),但是如果引入了数组,将计算结果预先存储在数组中,那么每次询问只需要 O(1) 的时间复杂度就可以得到第 n 项的值,而这时,由于引入了数组所以空间复杂度就变成了 O(n)。

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

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

相关文章

FFmpeg-chapter3-读取视频流(原理篇)

ffmpeg网站&#xff1a;About FFmpeg 1 库介绍 &#xff08;1&#xff09;libavutil是一个包含简化编程函数的库&#xff0c;包括随机数生成器、数据结构、数学例程、核心多媒体实用程序等等。 &#xff08;2&#xff09;libavcodec是一个包含音频/视频编解码器的解码器和编…

面试(进阶) —虚拟列表在什么场景使用,如何实现?

面试(进阶) —虚拟列表在什么场景使用&#xff0c;如何实现&#xff1f; 在前端开发中&#xff0c;当需要渲染大量数据时&#xff0c;传统的渲染方式往往会遇到性能瓶颈。一次性将大量数据渲染到DOM中&#xff0c;不仅会导致页面加载缓慢&#xff0c;还可能占用大量内存&#x…

Linux Mem -- 关于AArch64 MTE功能的疑问

目录 1.虚拟地址和物理地址映射完成后&#xff0c;才可以设置虚拟地址对应的memory tag &#xff1f; 2.各种memory allocator中的address tag从哪来&#xff0c;怎么产生&#xff1f; 2.1 vmalloc allocator 2.2 slub分配器 2.3 用户可以指定IRG指令产生的address tag 3.kasan…

python-leetcode-颜色分类

75. 颜色分类 - 力扣&#xff08;LeetCode&#xff09; class Solution:def sortColors(self, nums: List[int]) -> None:"""Do not return anything, modify nums in-place instead."""low, mid, high 0, 0, len(nums) - 1while mid < h…

ArcGIS Pro技巧实战:高效矢量化天地图地表覆盖图

在地理信息系统&#xff08;GIS&#xff09;领域&#xff0c;地表覆盖图的矢量化是一项至关重要的任务。天地图作为中国国家级的地理信息服务平台&#xff0c;提供了丰富且详尽的地表覆盖数据。然而&#xff0c;这些数据通常以栅格格式存在&#xff0c;不利于进行空间分析和数据…

【江科大STM32】TIM输出比较(学习笔记)

本章图片文字内容也为重要知识&#xff0c;请马住&#xff01; 输出比较简介 OC&#xff08;Output Compare&#xff09;输出比较输出比较可以通过比较CNT与CCR寄存器值的关系&#xff0c;来对输出电平进行置1、置0或翻转的操作&#xff0c;用于输出一定频率和占空比的PWM波形…

【网络安全 | 漏洞挖掘】利用文件上传功能的 IDOR 和 XSS 劫持会话

未经许可,不得转载。 本文涉及漏洞均已修复。 文章目录 前言正文前言 想象这样一个场景:一个专门处理敏感文档的平台,如保险理赔或身份验证系统,却因一个设计疏漏而成为攻击者的“金矿”。在对某个保险门户的文件上传功能进行测试时,我意外发现了一个可导致大规模账户接管…

飞算 JavaAI 如何让微服务开发快人一步?

在当今竞争激烈的软件开发领域&#xff0c;微服务架构因其灵活性和可扩展性备受青睐。然而&#xff0c;微服务开发过程复杂&#xff0c;从需求分析到最终代码实现&#xff0c;每个环节都需要耗费大量时间和精力。飞算 JavaAI 的出现&#xff0c;犹如一道曙光&#xff0c;为开发…

Python—Excel全字段转json文件(极速版+GUI界面打包)

目录 专栏导读1、背景介绍2、库的安装3、核心代码4、完整代码(简易版)5、进阶版(GUI)总结专栏导读 🌸 欢迎来到Python办公自动化专栏—Python处理办公问题,解放您的双手 🏳️‍🌈 博客主页:请点击——> 一晌小贪欢的博客主页求关注 👍 该系列文章专栏:请点击——…

2025年光电科学与智能传感国际学术会议(ICOIS 2025)

重要信息 官网&#xff1a;www.ic-icois.org 时间&#xff1a;2025年3月14-16日 地点&#xff1a;中国-长春 简介 2025年光电科学与智能传感国际学术会议&#xff08;ICOIS 2025&#xff09;将于2025年3月14-16日在中国-长春隆重召开。会议将围绕“光学光电”、“智能传感”…

企业微信里可以使用的企业内刊制作工具,FLBOOK

如何让员工及时了解公司动态、行业资讯、学习专业知识&#xff0c;并有效沉淀企业文化&#xff1f;一份高质量的企业内刊是不可或缺的。现在让我来教你该怎么制作企业内刊吧 1.登录与上传 访问FLBOOK官网&#xff0c;注册账号后上传排版好的文档 2.选择模板 FLBOOK提供了丰富的…

YOLOv5 + SE注意力机制:提升目标检测性能的实践

一、引言 目标检测是计算机视觉领域的一个重要任务&#xff0c;广泛应用于自动驾驶、安防监控、工业检测等领域。YOLOv5作为YOLO系列的最新版本&#xff0c;以其高效性和准确性在实际应用中表现出色。然而&#xff0c;随着应用场景的复杂化&#xff0c;传统的卷积神经网络在处…

跟我学C++中级篇——定时器的设计

一、定时器 谈到定时器&#xff0c;理论上讲是各种语言和各种设计都无法避开的一个技术点。对于定时器来说&#xff0c;表面上就是一种时间间隔的处理约定&#xff0c;但对程序来说&#xff0c;可能就是设计层面、接口层面和库或框架以及系统应用的一个大集合。不同的系统&…

智能机器人加速进化:AI大模型与传感器的双重buff加成

Deepseek不仅可以在手机里为你解答现在的困惑、占卜未来的可能&#xff0c;也将成为你的贴心生活帮手&#xff01; 2月21日&#xff0c;追觅科技旗下Dreamehome APP正式接入DeepSeek-R1大模型&#xff0c;2月24日发布的追觅S50系列扫地机器人也成为市面上首批搭载DeepSeek-R1的…

PostgreSQL10 逻辑复制实战:构建高可用数据同步架构!

PostgreSQL10 逻辑复制实战&#xff1a;打造高可用数据同步架构&#xff01; 概述 PostgreSQL 10 引入了逻辑复制&#xff08;Logical Replication&#xff09;&#xff0c;为数据库高可用和数据同步提供了更灵活的选择。PostgreSQL 复制机制主要分为物理复制和逻辑复制两种&…

LVS+Keepalived高可用群集配置案例

以下是一个 LVSKeepalived 高可用群集配置案例&#xff1a; 1、环境准备 LVS 主调度器&#xff08;lvs1&#xff09;&#xff1a;IP 地址为 192.168.8.101&#xff0c;心跳 IP 为 192.168.4.101LVS 备调度器&#xff08;lvs2&#xff09;&#xff1a;IP 地址为 192.168.8.102…

原生家庭独立的艺术:找到自我与家庭的平衡点

原生家庭独立的艺术&#xff1a;找到自我与家庭的平衡点 &#x1f331; 引言 &#x1f308; 小林刚刚和父母结束了一次激烈的电话对峙。父母坚持认为他应该回到家乡工作&#xff0c;“这样我们也能照顾你”&#xff0c;而他则努力解释自己在大城市的职业规划。挂掉电话后&…

Java进阶——注解一文全懂

Java注解&#xff08;Annotation&#xff09;是一种强大的元数据机制&#xff0c;为代码提供了附加信息&#xff0c;能简化配置、增强代码的可读性和可维护性。本文将深入探讨 Java 注解的相关知识。首先阐述了注解的基础概念&#xff0c;包括其本质、作用以及核心分类&#xf…

DeepSeek 15天指导手册——从入门到精通 PDF(附下载)

DeepSeek使用教程系列--DeepSeek 15天指导手册——从入门到精通pdf下载&#xff1a; https://pan.baidu.com/s/1PrIo0Xo0h5s6Plcc_smS8w?pwd1234 提取码: 1234 或 https://pan.quark.cn/s/2e8de75027d3 《DeepSeek 15天指导手册——从入门到精通》以系统化学习路径为核心&…

【智能音频新风尚】智能音频眼镜+FPC,打造极致听觉享受!【新立电子】

智能音频眼镜&#xff0c;作为一款将时尚元素与前沿科技精妙融合的智能设备&#xff0c;这种将音频技术与眼镜形态完美结合的可穿戴设备&#xff0c;不仅解放了用户的双手&#xff0c;更为人们提供了一种全新的音频交互体验。新立电子FPC在智能音频眼镜中的应用&#xff0c;为音…