数据结构:时间复杂度与空间复杂度

news2024/11/14 15:18:54

目录

  • 算法效率
  • 时间复杂度
    • 大O渐进表示法
    • 时间复杂度计算案例
  • 空间复杂度
    • 空间复杂度案例
  • 复杂度算法题

算法效率

算法在编写成可执行程序后,运⾏时需要耗费时间资源和空间(内存)资源 。因此衡量⼀个算法的好坏,⼀般是从时间和空间两个维度来衡量的,即时间复杂度和空间复杂度。
时间复杂度主要衡量⼀个算法的运行快慢,⽽空间复杂度主要衡量⼀个算法运行所需要的额外空间。
在计算机发展的早期,计算机的存储容量很小。所以对空间复杂度很是在乎。但是经过计算机行业的迅速发展,计算机的存储容量已经达到了很高的程度。所以我们如今已经不需要再特别关注⼀个算法的空间复杂度,有时我们会用空间复杂度来分担时间复杂度。

时间复杂度

定义:在计算机科学中,算法的时间复杂度是⼀个函数式T(N),它定量描述了该算法的运行时间。时间复杂度是衡量程序的时间效率,那么为什么不去计算程序的运行时间呢?

  1. 因为程序运行时间和编译环境和运行机器的配置都有关系,比如同⼀个算法程序,用⼀个老编译器进⾏编译和新编译器编译,在同样机器下运行时间不同。
  2. 同⼀个算法程序,用⼀个老的低配置机器和新的高配置机器,运行时间也不同。
  3. 并且时间只能程序写好后测试,不能写程序前通过理论思想计算评估。

算法的时间复杂度是⼀个函数式T(N)到底是什么呢?这个T(N)函数式计算了程序的执行次数。假设每个语句编译的时间都一样,那程序语句执行的次数就与时间成正比,用次数就可以间接反映算法的时间。
假设同一个问题,算法a程序T(N)=N,算法b程序T(N)=N^2,那算法a就优于算法b。
举个例子:

// 请计算⼀下Func1中++count语句总共执⾏了多少
次?
void Func1(int N)
{
int count = 0;
for (int i = 0; i < N ; ++ i)//外循环n次
{
for (int j = 0; j < N ; ++ j)//内循环n次
{
++count;//一共循环n*n次
}
}
for (int k = 0; k < 2 * N ; ++ k)//循环2n次
{
++count;
}
int M = 10;
while (M--)//循环10次
{
++count;
}
}

每个循环的语句都标注在程序后面,得出T(N)如下:
有的会问那些int M等语句不算吗?但是那些算上的话一共就两次,与N^2和2N比太小了,这里就直接忽略了。
T(N)=N^2+2N+10;
实际中我们计算时间复杂度时,计算的也不是程序的精确的执行次数,精确执行次数计算起来还是很麻烦的(不同的⼀句程序代码,编译出的指令条数都是不⼀样的),计算出精确的执行次数意义也不大,因为我们计算时间复杂度只是想比较算法程序的增长量级,也就是当N不断变大时T(N)的差别,上面我们已经看到了当N不断变大时常数和低阶项对结果的影响很小,所以我们只需要计算程序能代表增长量级的大概执行次数,复杂度的表示通常使用大O的渐进表示法。

大O渐进表示法

大O符号(Big O notation):是用于描述函数渐进行为的数学符号。

  1. 时间复杂度函数式T(N)中,只保留最高阶项,去掉那些低阶项,因为当N不断变大时,低阶项对结果影响越来越小,当N无穷大时,就可以忽略不计了。
    用上面的列子来说:T(N)就可以写为N^2,后面的低阶项和常数就可以忽略了。
  2. 如果最高阶项存在且不是1,则去除这个项目的常数系数,因为当N不断变大,这个系数对结果影响越来越小,当N无穷大时,就可以忽略不计了。
    例如:2N^2 与 N^2是等价的,当N无穷大,系数2的影响可以忽略。
  3. T(N)中如果没有N相关的项⽬,只有常数项,⽤常数1取代所有加法常数。

时间复杂度计算案例

案例1:

void Func2(int N)
{
int count = 0;
for (int k = 0; k < 2 * N ; ++ k)//循环n次
{
++count;
}
int M = 10;
while (M--)//循环10次
{
++count;
}
printf("%d\n", count);
}

T(N)=2N+10;
最高阶是2N,后面的低阶项和常数项忽略,最高阶的的系数也可以忽略;
所以这段程序的时间复杂度就是:O(N)。

案例2:

// 计算Func3的时间复杂度?
void Func3(int N, int M)
{
int count = 0;
for (int k = 0; k < M; ++ k)//循环M次
{
++count;
}
for (int k = 0; k < N ; ++k)//循环N次
{
++count;
}
printf("%d\n", count);
}

T(N)=M+N;
当M<<N时:复杂度为O(N);
当M>>N时:复杂度为O(M);
当M==N时:复杂度为O(M+N);
因为他有两个影响次数的参数,分别是M和N,复杂度算的是情况最坏的情况,也就是M和N都趋于无穷,那他们两个共同决定运行次数;
所以复杂度就是O(M+N);

案例3:

// 计算Func4的时间复杂度?
void Func4(int N)
{
int count = 0;
for (int k = 0; k < 100; ++ k)//循环100次
{
++count;
}
printf("%d\n", count);
}

T(N)=100;
由大O渐进表示法第三条可得:
时间复杂度为O(1);

案例4:
功能,在源字符串中寻找子字符串(假设字符串长度趋于无穷)

// 计算strchr的时间复杂度?
const char * strchr ( const char* str, int character)
{
const char* p_begin = s;
while (*p_begin != character)
{
if (*p_begin == '\0')
return NULL;
p_begin++;
}
return p_begin;
}

第一种情况:
要找的字符串在源字符串前面:Y(N)=常数;时间复杂度:O(I);
第二种情况:
在中间位置:T(N)=N/2;时间复杂度:O(N);
第三种情况:
在末尾:T(N)=N;时间复杂度:O(N);
取最坏结果:时间复杂度为O(N);

案例5:

// 计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{
if(0 == N)
return 1;
return Fac(N-1)*N;
}

递归就相当与循环调用一个函数N次;
通过判断可知这个递归函数一共调用N次;
所以它的时间复杂度:O(N)。

空间复杂度

空间复杂度也是⼀个数学表达式,是对⼀个算法在运行过程中因为算法的需要额外临时开辟的空间。
空间复杂度不是程序占用了多少bytes的空间,因为常规情况每个对象大小差异不会很大,所以空间复杂度算的是变量的个数。
空间复杂度计算规则基本跟实践复杂度类似,也使用大O渐进表示法。
注意:函数运行时所需要的栈空间(存储参数、局部变量、⼀些寄存器信息等)在编译期间已经确定好了,因此空间复杂度主要通过函数在运⾏时候申请的额外空间来确定。

空间复杂度案例

案例1:

// 计算BubbleSort的时间复杂度?
void BubbleSort(int* a, int n)
{
assert(a);
for (size_t end = n//1次; end > 0; --end)
{
int exchange = 0;//2次
for (size_t i = 1//3次; i < end; ++i)
{
if (a[i-1] > a[i])
{
Swap(&a[i-1], &a[i]);
exchange = 1;
}
}
if (exchange == 0)
break;
}
}

由分析可知在运行时创建的临时变量有三个,T(N)=3;
所以空间复杂度为O(1);

案例2:

// 计算阶乘递归Fac的空间复杂度?
long long Fac(size_t N)
{
if(N == 0)
return 1;
return Fac(N-1)*N;
}

在c语言学习中,函数的使用要创建栈帧空间,前面分析了这个递归要运行N次这个函数,运行一次就创建一个函数栈帧空间,所以T(N)=N;
它的空间复杂度:O(N)。

复杂度算法题

轮转N个数据:
数组元素: 0 1 2 3 4
轮转1次:4 0 1 2 3
轮转2次:3 4 0 1 2

void rotate(int* nums, int numsSize, int k) {
while(k--)
{
int end = nums[numsSize-1];
for(int i = numsSize - 1;i > 0 ;i--)
{
nums[i] = nums[i-1];
}
nums[0] = end;
}
}

外循环K次,内循环N次。
可知一共循环KN次。
最欢结果,让数据轮转N次,即T(N)=N
N=N^2;
时间复杂度:O(N^2)。

现在出个问题,让它的时间复杂度降低。
思路1:刚开始提到可以用空间复杂度来分担时间复杂度。
代码实现:

void rotate(int* nums, int numsize, int k) {
int newnums[numsize];//创建大小为N的字符串
for(int i =0;i<numsize;i++)
{
newnums[i]=nums[(i+k)%numsize];
}
for(int i=0;i<numsize;i++)
{
nums[i]=newnums[i];
}

时间复杂度:T(N)=2N–> O(N);
空间复杂度:T(N)=N–>O(N)。

思路2:
在这里插入图片描述
如图数组里存放5个数,轮转2次。
先操作前半部分:交换第一个与第二个数据,如果前半部分数据较多(交换第二个跟倒数第二的数据,依次类推)
交换后数组变成:2 1 3 4 5;
然后相同的方法交换后半部分,
交换后数组:2 1 5 4 3;
然后交换整个数组,
交换后:3 4 5 1 2,
交换完成。
空间复杂度O(1),时间复杂度O(N)

void swap(int* nums,int left, int right)
{
	while (left < right)
	{
		int tmp = nums[left];
		nums[left] = nums[right];
		nums[right] = tmp;

		left++;
		right--;
	}
	
}
void rotate(int* nums, int numsize, int k) {
	
	k%= numsize;
	swap(nums, 0,k-1);
	swap(nums, k,numsize-1);
	swap(nums, 0,numsize-1);
}

这个思路不难,就是不容易想出来。它既节省了空间复杂度,还优化了时间复杂度。
----------------------------------------------分隔符
时间复杂度与空间复杂度就介绍完了,希望对各位看官有所帮助。
有错请在评论区指正,谢谢。

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

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

相关文章

通过sshd_config限制用户登录

在CentOS Stream或其他现代的Linux发行版中&#xff0c;你可能会发现传统的hosts.deny和 hosts.allow文件已经不存在或不被使用。这是因为随着时间的推移&#xff0c;系统的安全策略和网络管理工具已经发生了演变&#xff0c;许多系统管理员和发行版维护者选择使用更现代、更灵…

12. DataLoader的基本使用

DataLoader的基本使用 1. 为什么要使用DataLoader DataLoader对创建好的DataSet的样本取样进行了集成操作&#xff0c;非常方便对于后续网络训练、测试的数据集的选择和使用 DataLoader可以集成了数据批量加载的方法&#xff0c;可以使用 batch_size 设置批量大小&#xff0c…

深入理解 Redis 的文件事件处理器

概述 Redis 的文件事件处理器是基于 Reactor 模式实现的&#xff0c;内部采用 IO 多路复用程序来同时监听多个套接字&#xff0c;当被监听的套接字准备好执行连接应答&#xff08;accept&#xff09;、读取&#xff08;read&#xff09;、写入&#xff08;write&#xff09;、…

计算机毕业设计 高校大学生竞赛项目管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

获取Live2d模型

文章目录 1、 Live2D官方示例数据集&#xff08;可免费下载&#xff09;2、模之屋3、unity商店4、直接b站搜索5、youtube6、BOOTH完结 1、 Live2D官方示例数据集&#xff08;可免费下载&#xff09; 官方提供了一些 Live2D实例模型给大家下载使用 地址&#xff1a;https://ww…

2024年【山东省安全员B证】报名考试及山东省安全员B证最新解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 山东省安全员B证报名考试是安全生产模拟考试一点通生成的&#xff0c;山东省安全员B证证模拟考试题库是根据山东省安全员B证最新版教材汇编出山东省安全员B证仿真模拟考试。2024年【山东省安全员B证】报名考试及山东省…

通用接口开放平台设计与实现——(31)API服务线程安全问题确认与修复

背景 在本系列的前面一篇博客评论中&#xff0c;有小伙伴指出&#xff0c;API服务存在线程安全问题&#xff1a; https://blog.csdn.net/seawaving/article/details/122905199#comments_34477405 今天来确认下&#xff0c;线程是否安全&#xff1f;如不安全&#xff0c;如何…

在k8s中,客户端访问服务的链路流程,ingress--->service--->deployment--->pod--->container

ingress是一个API资源。 其核心作用是nginx网页服务器。 当客户端访问服务器不同的url时, 用不同的location提供服务。 在k8s之外&#xff0c;nginx的配置一般如下&#xff1a; http {server {listen 80;server_name localhost;location / {root html; …

文件的应用实例

目录 1、拷贝文件 2、遍历文件夹 1、拷贝文件 说明&#xff1a;将一张图片/一首歌拷贝到另外一个目录下&#xff0c;要求使用read()和write()原生方法完成 """思路分析&#xff1a;1、打开源文件(需要拷贝的文件)&#xff0c;读取源文件的数据2、打开目标文…

网络安全学习(四)渗透工具msf

本文简要介绍metasploit framework&#xff0c;是一款渗透工具。官网地址&#xff1a;Metasploit | Penetration Testing Software, Pen Testing Security | Metasploit msf是一个框架&#xff0c;可以加载各种模块&#xff0c;这是它的最强大之处。 kali中有此工具。 点击即…

python中的各类比较与计算

运算符 1.算数运算符2.关系运算符3.逻辑运算符4.关于短路求值5.赋值运算符1&#xff09;的使用链式赋值多元赋值 2)复合赋值运算符 6.位运算符7.成员运算符8.身份运算符 1.算数运算符 # 加 print(1 2) # 减 print(2 - 1) # 乘 print(1 * 2) # 余数 4%31余数为1 print(4 % 3…

C++第五十一弹---IO流实战:高效文件读写与格式化输出

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】 目录 1. C语言的输入与输出 2. 流是什么 3. CIO流 3.1 C标准IO流 3.2 C文件IO流 3.2.1 以写方式打开文件 3.2.1 以读方式打开文件 4 stringstre…

【测试方案】软件测试管理规程(doc源文件)

软件测试规程的作用在于确保软件测试活动的系统性、规范性和一致性。它明确了测试的目标、范围、方法、流程以及所需资源&#xff0c;为测试人员提供了明确的指导和操作规范。通过遵循测试规程&#xff0c;可以提高测试效率&#xff0c;减少测试遗漏和错误&#xff0c;保证软件…

NC 表达式求值

系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂了就去分享给你的码吧。 描述 请写一个整数…

MySQL8.0.13-函数索引

目录 什么是函数索引 语法 函数索引测试 创建表结构 插入数据 创建普通索引 查看执行计划 创建函数索引 查看执行计划 查看索引信息 函数索引效率 普通索引 函数索引 分析 注意事项 老版本如何实现函数索引 什么是函数索引 本篇主要介绍 MySQL 的函数索引&…

【数据结构】顺序表和链表经典题目

系列文章目录 单链表 动态顺序表实现通讯录 顺序表 文章目录 系列文章目录前言一、顺序表经典例题1. 移除元素2. 合并两个有序数组 二、链表经典例题1. 移除链表元素2. 反转链表3. 合并两个有序链表4. 链表的中间节点5. 环形链表的约瑟夫问题 总结 前言 我们通过前面对顺序表…

NVM(node.js版本工具)的使用

1.nvm是什么 NVM 是 Node Version Manager 的缩写&#xff0c;它是一个用于管理 Node.js 版本的命令行工具。通过NVM&#xff0c;你可以在同一台机器上安装和切换多个 Node.js 版本&#xff0c;对于开发和测试在不同 Node.js 版本上运行的应用程序非常有用。 2.下载 下载之前…

『功能项目』眩晕图标显示【52】

我们打开上一篇51调整Boss技能bug的项目&#xff0c; 本章要做的事情是在释放法师的眩晕技能时&#xff0c;boss01处在眩晕动画时显示一个眩晕图标 首先双击Boss01预制体进入预制体空间 创建一个Image重命名为StateUIdiz 代表第一个受击状态 设置Canavas 并且修改Canvas的渲染…

Java 学习全攻略:从入门到精通的详细指南

目录 一、引言 Java 的背景和发展 学习 Java 的意义 二、Java 的核心特性 1. 面向对象编程&#xff08;OOP&#xff09; 2. 跨平台性 3. 自动内存管理 4. 强大的标准库 三、Java 基础语法 1. 变量和数据类型 原始数据类型 引用数据类型 2. 运算符 3. 控制结构 条…

柳淘鸿黄金沁透发热面膜:肌肤逆龄之旅的秘密武器!

柳淘鸿黄金沁透发热面膜&#xff1a;肌肤逆龄之旅的秘密武器&#xff01;"柳淘鸿的黄金沁透发热面膜液融合了中国发明专利,专利号:ZL202310228041.5对应成分:胶原, 金&#xff0c;珍珠粉以及多种珍贵植物萃取精华&#xff0c;是肌肤逆龄之旅的绝密武器。这款面膜液温和滋养…