数据结构之时间复杂度-空间复杂度

大家好,我是深鱼~

目录

1.数据结构前言

1.1什么是数据结构

1.2什么是算法

1.3数据结构和算法的重要性

1.4如何学好数据结构和算法

2.算法的效率

3.时间复杂度

3.1时间复杂度的概念

3.2大O的渐进表示法

【实例1】:双重循环的时间复杂度:O(N)

【实例2】:双重循环的时间复杂度:O(N+M)

【实例3】:常数循环的时间复杂度:O(1)

【实例4】:strchr的时间复杂度:O(N)

【实例5】:冒泡排序的时间复杂度:O(N^2)

【实例6】:二分查找的时间复杂度:O(log2N)

【实例7】:阶乘递归的时间复杂度:O(N)

【实例8】:斐波那契递归的时间复杂度:O(2^N)

 4.空间复杂度

【实例1】:冒泡排序的空间复杂度:O(1)

【实例2】:斐波那契递归的空间复杂度:O(N)

【实例3】:函数阶乘递归的空间复杂度:O(N)

 【拓展】递归版斐波那契数列的空间复杂度:O(N)


1.数据结构前言

1.1什么是数据结构

实现一些项目,需要在内存中将数据存储起来,数据结构就是计算机存储、组织数据的方式。指相互之间存在一种或多种特定关系的数据元素的集合。eg:数组,链表,树...

1.2什么是算法

算法简单来说就是一系列的计算步骤,用来将输入数据转化为输出结果的。常见的算法有:排序,查找,查重,推荐算法...

1.3数据结构和算法的重要性

在校招的笔试中会有很多有关数据结构和算法的题

可以看看链接,在未来工作中:

数据结构和算法对一个程序员来说的重要性

1.4如何学好数据结构和算法

<1>多敲代码

<2>注重画图思考

2.算法的效率

算法的效率看两点,第一点看时间效率,也就是时间复杂度,第二点看空间效率,也就是空间复杂度,但是随着计算机行业的发展,计算机的存储容量已经达到了很高的程度,所以如今我们不用太关注一个算法的空间复杂度

3.时间复杂度

3.1时间复杂度的概念

算法的时间复杂度是数学里面一个带有未知数的函数表达式,算法的复杂度不是看这个算法的运行时间,因为环境不同,具体的运行时间就不一样,eg:10年前2核cpu、2g内存的机器和今天8核cpu、8g内存的机器,运行的时间就不一样。算法中的基本操作的执行次数,为算法的时间复杂度

3.2大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--)
{++count;
}
printf("%d\n", count);
}

Func1 执行的基本操作次数 :F(N)=N*N+2*N+10

当N = 10        F(N) = 130

当N = 100      F(N) = 10210

当N = 1000   F(N) = 1002010

N越大,后两项对结果的影响越小,所以实际计算时间复杂度时,我们只需要大概执行次数,那么这里我们使用大O的渐进表示法(估算),即时间复杂度:O(N^2)

大O渐进表示法:

(1)用常数1取代运行时间中的所有加法常数

(2)在修改后的运行次数函数中,只保留最高阶项

(3)如果最高阶存在且不是1,则取除与这个项目相乘的常数

【实例1】:双重循环的时间复杂度:O(N)

本来应该是2*N,根据大O渐进表示法(3)简化为O(N)

// 计算Func2的时间复杂度?
void Func2(int N)
{
int count = 0;
for (int k = 0; k < 2 * N ; ++ k)
{
++count;
}
int M = 10;
while (M--)
{
++count;
}
printf("%d\n", count);
}

【实例2】:双重循环的时间复杂度:O(N+M)

(如果前提:M>>N,那么时间复杂度就是O(M);

                      N>>M,那么时间复杂度就是O(N);

                      M和N差不多,那么时间复杂度O(M)或O(N)都可以)

一般情况下时间复杂度计算时未知数都是用的N,但是也可以使用M,K等等其他的

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

【实例3】:常数循环的时间复杂度:O(1)

本来是100,根据大O渐进表示法(1)简化为O(1)

O(1)不是代表算法运行一次,而是常数次

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

【实例4】:strchr的时间复杂度:O(N)

// 计算strchr的时间复杂度?
const char * strchr ( const char * str, int character );

strchr函数的逻辑实际就是下面这个

while(*str)

{

     if(*str==character)

            return str;

     else

           ++str;

}

 以hello world这个字符串为例:

假设查找的是h:      1 最好情况:任意输入规模的最小运行次数(下界)

假设查找的是w:     N/2 平均情况:任意输入规模的期望运行次数(大概就是最好最坏相加/2)

假设查找的是d:       N 最坏情况:任意输入规模的最大运行次数(上界)

当一个算法随着输入的不同,时间复杂度不同,时间复杂度做悲观预期,看最坏的情况(即这个例子的时间复杂度是O(N))

【实例5】:冒泡排序的时间复杂度:O(N^2)

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

时间复杂度:N-1,N-2,N-3...1   精确值也就是N*(N-1)/2,那么大O的渐变表示法就是O(N^2)

算时间复杂度不能只看几层循环,而要去看他的思想

【实例6】:二分查找的时间复杂度:O(log2N)

// 计算BinarySearch的时间复杂度?
int BinarySearch(int* a, int n, int x)
{
assert(a);
int begin = 0;
int end = n-1;
while (begin < end)
{
int mid = begin + ((end-begin)>>1);
if (a[mid] < x)
begin = mid+1;
else if (a[mid] > x)
end = mid;
else
return mid;
}
return -1;
}

最好的情况:O(1)

最坏的情况:O(log2N)

为什么是O(log2N)呢?

【画图理解】:假设我们要查找X次,一个数组的大小是N,每一次二分查找如果没有找到,N就除以2,考虑最坏的结果,那么直到N一直除到只剩1为止就结束了

N/2/2/2/2...=1

2^X=N

X=log2N

 可见二分查找算法是一个非常牛逼的算法

N个数中查找                大概查找次数

1000                              10

100W                             20

10亿                              30

但是这个算法的前提是数组有序

【实例7】:阶乘递归的时间复杂度:O(N)

递归算法时间复杂度:递归次数*每次递归调用的次数

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

Fac(N)   Fac(N-1)  ... Fac(1)

【实例8】:斐波那契递归的时间复杂度:O(2^N)

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

【画图理解】:理解递归的逻辑思想,每一次递归都会调用小的两个递归,最后右边的递归调用会先结束,那么递归的次数就是等比数列的和减去右下角因提前结束而缺少的次数

Fib(N)=2^0+2^1+2^2+...+2^n-X

此处的每次递归调用的次数是个常数,就相当于没*

那么大O渐进表示法也就是O(2^N)

可见斐波那契数列的递归写法完全是一个没有实际用途的算法,因为太慢了

 4.空间复杂度

空间复杂度也是一个数学表达式,是一个算法在运行过程中的临时额外占用存储空大小的量度

空间复杂度不是程序占用了多少bytes的空间,因为这个也没有太大的意义,所以空间复杂度算的是变量的个数

空间复杂度计算规则基本跟时间复杂度类似,也使用大O渐进表示法

【注意】:函数运行时所需要的栈空间(存储参数,局部变量,一些存储器信息等等)在编译期间已经确定好了,因此空间复杂度主要通过函数在运行时申请额外空间来确定

【实例1】:冒泡排序的空间复杂度:O(1)

冒泡排序中有三个变量:exchang,end,i,那么根据大O渐进表示法为O(1)

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

【实例2】:斐波那契递归的空间复杂度:O(N)

N个数的数组,动态开辟了N+1个空间,简化过后空间复杂度为O(N)

这个函数返回的是斐波那契数列的前n项的数组,而不是一个数

那个函数的时间复杂度为O(N),比递归的O(2^N)简化了很多

// 计算Fibonacci的空间复杂度?
//返回斐波那契数列的前n项
long long* Fibonacci(size_t n)
{
if(n==0)
return NULL;
long long * fibArray =
(long long *)malloc((n+1) * sizeof(long long));
fibArray[0] = 0;
fibArray[1] = 1;
for (int i = 2; i <= n ; ++i)
{
fibArray[i ] = fibArray[ i - 1] + fibArray [i - 2];
}
return fibArray ;
}

【实例3】:函数阶乘递归的空间复杂度:O(N)

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

【画图理解】:递归函数调用了N次,开辟了N个栈帧,每个栈帧使用了常数的个空间,所以空间复杂度为O(N) (只要看递归的深度

 【拓展】递归版斐波那契数列的空间复杂度:O(N)

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

【画图理解】: 本函数调用空间的顺序是Fbi(N),Fbi(N-1)...Fbi(1),也就是最左边的一个枝干,然后这些函数的空间销毁,继续下一个枝干,这样函数递归的深度一直都是N,而不会是2^N

空间是可以重复利用,不累计的

时间是一去不复返,累积的

这次数据结构之时间和空间复杂度的内容就到此啦,有什么问题欢迎评论区或者私信交流,觉得笔者写的还可以,或者自己有些许收获的,麻烦铁汁们动动小手,给俺来个一键三连,万分感谢 ! 

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

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

相关文章

一文解决JWT相同签名不匹配问题【JWT signature does not match locally computed signature.】

今天做项目的时候&#xff0c;涉及到一个支付记账的功能&#xff0c;想着不能将这些金额数据显示暴露的通过常规的请求体封装来进行传输&#xff0c;想着要是被中途抓包修改了不就麻烦了&#xff0c;所以考虑到这种安全性的需求&#xff0c;就利用上了JWT来进行数据的封装传递&…

Dubbo基于springboot学习笔记

本文参考&#xff1a;【优极限】最透彻的Dubbo教程&#xff08;dubbo经典之作完整版&#xff09;&#xff0c;阿里分布式框架dubbo零基础实战教学_手把手地啊你读懂底层源码【完整版】_哔哩哔哩_bilibili 1、 互联网架构演变 &#xff08;1&#xff09;单一应用架构 把系统中…

AWS——03篇(AWS之Amazon S3(云中可扩展存储)-01入门)

AWS——03篇&#xff08;AWS之Amazon S3&#xff08;云中可扩展存储&#xff09;-01入门&#xff09; 1. 前言2. 关于 Amazon S32.1 介绍2.1.1 简述2.1.2 详细介绍 2.2 Amazon S3 好处和功能2.3 3. 创建S3存储桶3.1 创建存储桶3.2 修改访问权限 4. 简单实用4.1 上传图片文件4.2…

2023年深度学习最新研究成果

LLMs领域 AGI领域 无剑芯片设计平台 三级标题 四级标题 五级标题 六级标题

电脑选购:6000元左右买到性价比超高的笔记本电脑,准大学生的购机指南

目录 一、ThinkBook 14 二、华硕灵耀14 2023 四、宏碁掠夺者擎Neo 五、惠普&#xff08;HP&#xff09;暗影精灵9 六、联想拯救者R7000P 2023 每年高考毕业季&#xff0c;许多即将进入大学的毕业生都会面临新电脑的选择&#xff0c;而对于喜欢玩游戏的同学&#xff0c;一般…

vscode extension 怎么区分dev prod

开发模式注入环境变量 使用vsode 提供的api

实现跨域的几种方式

原理 前后端的分离导致了跨域的产生 跨域的三要素&#xff1a;协议 域名 端口 三者有一个不同即产生跨域 例如&#xff1a; http://www.csdn.com https://www.csdn.com 由于协议不同&#xff0c;端口不同而产生跨域 注&#xff1a;http的默认端口80&#xff0c;https的默…

【云原生】kubernetes在Pod中init容器的作用和使用

目录 Pod 中 init 容器 1 init 容器特点 2 使用 init 容器 Pod 中 init 容器 Init 容器是一种特殊容器&#xff0c;在Pod 内的应用容器启动之前运行。Init 容器可以包括一些应用镜像中不存在的实用工具和安装脚本。 1 init 容器特点 init 容器与普通的容器非常像&#xf…

ffmpeg 4.4版本对MP4文件进行AES-CTR加密,和流式加密

对于ffmpeg的AES-CTR加密有两种方式&#xff0c;一个是普通的整个视频做加密&#xff0c;另一个是对视频做切片处理&#xff0c;然后进行加密。 一、对于普通的加密方式 直接使用下面的命令就行 ffmpeg -i animal.mp4 -vcodec copy -acodec copy -encryption_scheme cenc-aes…

本地 shell无法连接centos 7 ?

1、首先检查是否安装ssh服务&#xff1b; yum list installed | grep openssh-server# 没有安装尝试安装下 yum install openssh-server 2、检查ssh服务是否开启 systemctl status sshd.service# 未开启&#xff0c;开启下 systemctl start sshd.service # 将sshd 服务添…

STM32F103——GPIO八种工作模式

目录 1、GPIO 基本结构分析 2、GPIO 八种工作模式 2.1 输入浮空 2.2 输入上拉 2.3 输入下拉 2.4 模拟功能 2.5 开漏输出 2.6 开漏式复用功能 2.7 推挽输出 2.8 推挽式复用功能 3、GPIO 八种工作模式特点及应用 1、GPIO 基本结构分析 STM32F103的 GPIO 工作有八种…

24届近5年南京理工大学自动化考研院校分析

今天学长给大家带来的是南京理工大学控制考研分析 满满干货&#xff5e;还不快快点赞收藏 一、南京理工大学 ​ 学校简介 南京理工大学是隶属于工业和信息化部的全国重点大学&#xff0c;学校由创建于1953年的新中国军工科技最高学府——中国人民解放军军事工程学院&#xf…

Android OkHttp源码分析--分发器

OkHttp是当下Android使用最频繁的网络请求框架&#xff0c;由Square公司开源。Google在Android4.4以后开始将源码中 的HttpURLConnection底层实现替换为OKHttp&#xff0c;同时现在流行的Retrofit框架底层同样是使用OKHttp的。 OKHttp优点: 1、支持Http1、Http2、Quic以及Web…

华秋亮相2023世界汽车制造技术暨智能装备博览会,推动汽车产业快速发展

洞悉全球汽车产业格局&#xff0c;前瞻业界未来趋势。2023年7月27日-30日&#xff0c;时隔三年&#xff0c;重聚武汉国际博览中心&#xff0c;2023世界汽车制造技术暨智能装备博览会盛大开幕。深耕汽车行业多年的世界汽车制造技术暨智能装备博览会&#xff0c;掀起行业热点新高…

boost beast http server 测试

boost beast http client boost http server boost beast 是一个非常好用的库&#xff0c;带上boost的协程&#xff0c;好很多东西是比较好用的&#xff0c;以下程序使用四个线程开启协程处理进入http协议处理。协议支持http get 和 http post #include <boost/beast/cor…

比较研发项目管理系统:哪个更适合您的需求?

项目管理系统对于保持项目进度、提高效率和确保质量至关重要。然而&#xff0c;市场上众多的研发项目管理系统让许多团队陷入选择困难。本文将对几个主流的研发项目管理系统进行深入分析&#xff0c;以帮助您找到最适合您团队的解决方案。 “哪个研发项目管理系统好用好&#x…

【Linux】常用的文本处理命令详解 + 实例 [⭐实操常用,建议收藏!!⭐]

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; &#x1f40b; 希望大家多多支…

【iPhone】手机还有容量,拍视频却提示 iPhone 储存空间已满

文章目录 前言解决方案 结语 前言 今天在用 iPhone 录像的时候突然提醒我 iPhone储存空间已满 你没有足够的储存空间来录制视频” 可我明明还有 20G 的容量 我非常疑惑&#xff0c;因为我之前还剩1个G都能录像&#xff0c;现在20G反而不行了&#xff0c;于是重启了手机&#…

怎么在JMeter中的实现关联

我们一直用的phpwind这个系统做为演示系统, 如果没有配置好的同学, 请快速配置之后接着往下看哦. phpwind发贴时由于随着登陆用户的改变, verifycode是动态变化的, 因此需要用到关联. LoadRunner的关联函数是reg_save_param, Jmeter的关联则是利用后置处理器来完成. 在需要查…

在线高精地图生成算法调研

1.HDMapNet 整体的网络架构如图所示&#xff0c;最终的Decoder输出三个分支&#xff0c;一个语义分割&#xff0c;一个embedding嵌入分支&#xff0c;一个方向预测。然后通过后处理将这些信息处理成向量化的道路表示。 img2bev的方式之前有IPM&#xff0c;通过假设地面的高度都…