数据结构开始——时间复杂度和空间复杂度知识点笔记总结
好了,经过了漫长的时间学习c语言语法知识,现在我们到了数据结构的学习。
首先,我们得思考一下
什么是数据结构?
数据结构(Data Structure)是计算机存储、组织数据的方式,指相互之间存在一种或多种特定关系的数据元素的集合。
算法又是什么?
定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为输出。简单来说算法就是一系列的计算步骤,用来将输入数据转化成输出结果
算法讲究效率
如何衡量一个算法的好坏
如,斐波那契数列(虽然代码简单,但是它的效率高吗?)
long long Fib(int n)
{
if(n < 3)
{return 1;
}return Fib(n-1) + Fib(n-2);
}

是先(黑色)一步一步进去下去,算到Fib(1)时,再一步一步返回上去(红色)
这得花费非常多的时间,效率并不高。
因此,要衡量一个算法的好坏,一般是从时间和空间两个维度来衡量的即时间复杂度,空间复杂度
时间复杂度和空间复杂度
时间复杂度和空间复杂度的区别
作用:时间复杂度主要衡量一个算法的运行快慢,而空间复杂度主要衡量一个算法运行所需要的额外空间(由于目前的空间都很大,基本不用担心空间的)
时间复杂度
时间复杂度:算法的时间复杂度是一个函数
一个算法所花费的时间与其中语句的执行次数成正比例,算法中的基本操作的执行次数,为算法的时间复杂度也就是:找到某条基本语句与问题规模N之间的数学表达式,就是算出了该算法的时间复杂度。
请计算一下Func1中++count语句总共执行了多少次?
void Func1(int N)
{
int count = 0;
for (int i = 0; i < N ; ++ i)
{
for (int j = 0; j < N ; ++ j)
{ 这里是N^2
++count;
}
}
for (int k = 0; k < 2 * N ; ++ k)
{
++count; 这里是2*N
}
int M = 10;
while (M--)
{ 这里是10次
++count;
}
经过计算我们可以得出:是F(N)=N^2+2*N+10
其实,对于求复杂度,我们一般使用大O的渐进表示法,,现在来介绍下:
大O的渐进表示法
1.大O符号:是用于描述函数渐进行为的数学符号
2.规则方法:
1)用常数1取代运行时间中的所有加法常数。
2)、在修改后的运行次数函数中,只保留最高阶项。
3)、如果最高阶项存在且不是1,则去除与这个项目相乘的常数。
最后得到的结果就是大O阶
eg:N(10000000000000)这应该写成什么呢?
答案:仍然是O(1).
有人会问为啥它这个数那么大了,还是用O(1)表示呢?
现在我们来用形象的例子来解释下:
假如那里有座泰山,我们要铲平,
不同视角:会不同
我们人的视角觉得它非常大,
但是放到太阳系里面,就显得非常小了。同样,对于编程也是一样的道理。
通过上面我们会发现大O的渐进表示法去掉了那些对结果影响不大的项,简洁明了的表示出了执行次数
另外有些算法的时间复杂度存在最好、平均和最坏情况:
最坏情况:任意输入规模的最大运行次数(上界)平均情况:任意输入规模的期望运行次数
最好情况:任意输入规模的最小运行次数(下界)例如:在一个长度为N数组中搜索一个数据x
最好情况:1次找到最坏情况:N次找到平均情况:N/2次找到
所以,我们实际中一般关注的是最坏的情况。
原因:同样是以一个形象的例子来解释
约会的预期管理
情人节约会可能到时的时间
最好:17:00
平均:18:00
最坏:19:00
那么因为只有30%的机会准时到
所以是不是选择说19:00时最好,把预期拉到最低。
常见时间复杂度计算举例
void Func2(int N)
{
int count = 0;
for (int k = 0; k < 2 * N ; ++ k)
{ 这里是2*N次
++count;
}
int M = 10;
while (M--) 这里是10次
{
++count;
}
printf("%d\n", count);
}
所以,函数是不是就是:F(N)=2*N+10
根据大O法:得时间复杂度就是O(N)
例子2:
// 计算Func3的时间复杂度?
void Func3(int N, int M)
{
int count = 0;
for (int k = 0; k < M; ++ k)
{
++count; 这里是M次
}
for (int k = 0; k < N ; ++ k)
{ 这里是N次
++count;
}
printf("%d\n", count);
}
函数式:F(N)=M+N
注意:这里的M,N都是未知数,并不是具体数
所以根据大O法:时间复杂度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);
}
函数式:F(N)=100
根据大O法:由于它是常数嘛,所以用1代替
所以时间复杂度:O(1).
例子4:
const char * strchr ( const char * str, int character );
对于指针,如果没有具体说,一般默认都是O(N)
例子5:
void BubbleSort(int* a, int n)
{
assert(a);
for (size_t end = n; end > 0; --end){ 一般没说的话,默认N次int exchange = 0;for (size_t i = 1; i < end; ++i) 因为两个for循环,所以是N^2次{if (a[i-1] > a[i]){Swap(&a[i-1], &a[i]);exchange = 1;}}if (exchange == 0) 因为这里有了判断,它可以改变最好的情况break; 如果没有这句,} 最好也是O(N^2)因为它无法知道它已经排序了 }
排序
最好(已经排序好大小):O(1) X有人说是看一眼就知道它的大小排序了,但是你一亿的时候呢,你并不知道其他数字2与这个的大小?
你是不是也是要一个一个对比下去呢?
所以 时间复杂度: O(N) √最坏:O(N^2)
其实这里不敢数循环,因为以后会学更复杂的
N-1 N-2 N-3....3 2 1 等差数列 算出O(N^2)
例子5:我们之前写过的二分查找
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;elsereturn mid;
}return -1;
}
四 第三次 第二次 第一次
我们可以发现,这是不是一次查找,我们就可以筛选掉一半。
最坏情况:区间缩放到一个值时,要么找到,要么找不到
假设N是数组个数,x是最坏查找次数
N/2/2/2/2.../2=1
折半查找多少次就除多少个2
假设是x次2^x=N
x=log N
这里我们先来说明一下:

接着,我们来看一下二分查找的显著优势:
对比:
N 1000 100W 10亿 2^10=1024
O(N) 1000 100W 10亿 2^20=1024*1024
O(log N) 10 20 30 2^30=1024*1024*1024我们可以看到这二分查找极大改变它的时间效率。
但是一个很不友好的一点是,它仅仅局限于排序好的数,所以我们以后也不太实用
例子6:
计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{
if(0 == N)
return 1;
return Fac(N-1)*N;
}

这是不是相当于有N+1次,O(N+1)
根据大O法:时间复杂度:O(N)
Fac(N)
Fac(N-1)
Fac(N-2)
....
Fac(1)
Fac(0)
例子七:
// 计算斐波那契递归Fib的时间复杂度?
long long Fib(size_t N)
{
if(N < 3)
return 1;
return Fib(N-1) + Fib(N-2);
}

等比数列 -缺失(常数)
时间:O(2^N)
空间:O(N)对此,我们可以得出来:
时间一去不复返,不可重复利用
空间用了以后要归还,可以重复利用。
空间复杂度
1.空间复杂度也是一个数学表达式,是对一个算法在运行过程中临时占用存储空间大小的量度
2.空间复杂度算的是变量的个数(而不是程序占用了多少bytes的空间,因为这个也没太大意义)
注意:函数运行时所需要的栈空间(存储参数、局部变量、一些寄存器信息等)在编译期间已经确定好了,因此空间复杂度主要通过函数在运行时候显示申请的额外空间来确定。
3.空间复杂度的使用规则也是:大O法
空间复杂度举例:
例子1:
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;}
}
我们上面说了空间复杂度算的是变量的个数
我们看到了上面用了三个变量:end,exchange,i。 -->O(3)
根据大O法:O(1).
例子2:绝大多数空间复杂度都是O(1)或O(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; 为什么要多加1,因为数组从0开始,fibArray[1] = 1; for (int i = 2; i <= n ; ++i){fibArray[i] = fibArray[i - 1] + fibArray [i - 2];}return fibArray;
}
上面创建了n+1个空间
所以根据大O法:O(N)
例子3:
// 计算阶乘递归Fac的空间复杂度?
long long Fac(size_t N)
{
if(N == 0)
return 1;
return Fac(N-1)*N;
}
值得注意的是
递归调用了N次,开辟了N个栈帧,每个栈帧使用了常数个空间
函数在栈的开辟函数栈帧时是单独开辟的。调用本身时,会再2开辟一块空间作为新的函数栈帧,但是在此外并没有创建额外的变量,所以会认为创建的每一块栈帧空间复杂度变量是常数个。
举个形象例子:
空间销毁是把那个使用权还给操作系统了,并不是说这块空间就没了
内存的申请,就相当于 住酒店一样,当这房间不用了,这不是说把它炸所以参数从N到0,共递归了N+1次,
根据大O法:O(N)
例子6:
同样,斐波那契数列也是一样。
long long Fib(size_t N)
{
if(N < 3)
return 1;
return Fib(N-1) + Fib(N-2);
}
空间销毁是把那个使用权还给操作系统了,我们上面说过空间是可以重复使用的,
调用某个函数结束后,再次调用此函数,所用的空间与上一次的函数空间是同一个。
所以,同样是递归了N-1次
由大O法:O(N)。
大O的常见的变化表

祝福语:
最后的最后,希望我们都能战胜困难,一步一步地坚持下去,为以后拿到好offer!
相关文章:
数据结构开始——时间复杂度和空间复杂度知识点笔记总结
好了,经过了漫长的时间学习c语言语法知识,现在我们到了数据结构的学习。 首先,我们得思考一下 什么是数据结构? 数据结构(Data Structure)是计算机存储、组织数据的方式,指相互之间存在一种或多种特定关系的数据元素…...
路由策略与策略路由
路由策略 常用有Router-Policy,Filter-Policy等 控制路由是否可达,通过修改路由条目相关参数影响流量的转发 基于控制平面,会影响路由表表项,但只能基于目地址进行策略判定,于路由协议相结合使用 Router-Policy …...
pytorch_fid 安装笔记
目录 torch安装: pytorch_fid安装 torch安装: pip install torch2.5.0 --index-url https://download.pytorch.org/whl/cu121 pytorch_fid安装 pip install pytorch_fid 安装后,torch也会自动安装,导致torch引用报错。...
Qt绘制仪表————附带详细说明和代码示例
文章目录 1 效果2 原理3 编码实践3.1 创建仪表属性类3.2 设置类属性3.3 绘制图案3.3.1 设置反走样3.3.2 绘制背景3.3.3 重新定义坐标原点3.3.4 绘制圆环3.3.5 绘制刻度线3.3.6 绘制刻度线上的描述值3.3.7 绘制指针3.3.8 绘制指针数值和单位3.3.9 控制指针变化 扩展福利参考 1 效…...
百度地图JavaScript API核心功能指引
百度地图JavaScript API是一套由JavaScript语言编写的应用程序接口,它能够帮助您在网站中构建功能丰富、交互性强的地图应用,包含了构建地图基本功能的各种接口,提供了诸如本地搜索、路线规划等数据服务。百度地图JavaScript API支持HTTP和HT…...
mp4影像和m4a音频无损合成视频方法
第一步:复制高清视频地址 url 第二步:打开网址粘贴复制的视频url视频下载 第三步:下载-影像.mp4和-音频.m4a 第四步:合并视频; 使用ffmpeg进行无损合成(如果没有安装ffmpeg请自行下载安装下载 FFmpeg (p2hp.com)&…...
Ubuntu下将Julia嵌入Jupyter内核
一.安装 Julia 如果 Julia 尚未安装: 打开终端,下载最新的 Julia 安装包: wget https://julialang-s3.julialang.org/bin/linux/x64/1.9/julia-1.9.3-linux-x86_64.tar.gz 解压并移动到 /opt: tar -xvzf julia-1.9.3-linux-x86_…...
openGauss开源数据库实战二十五
文章目录 任务二十五 openGauss 数据库的物理备份与恢复任务目标实施步骤一、为进行物理备份做准备1.确保数据库工作在归档模式2.创建保存数据库物理备份的目录3.创建保存归档日志备份的目录 二、进行openGauss数据库的物理备份1.备份数据库2.切换WAL3.备份归档日志 三、openGa…...
[C/C++] List相关操作
List相关操作 1 链表二分 目标: (1)对于偶数节点,正好对半分; (2)对于奇数节点,前 后 1 (3)断开链表,方便后期合并 // 使用快慢指针完成中点…...
继电器控制与C++编程:实现安全开关控制的技术分享
在现代生活中,继电器作为一种重要的电气控制元件,在电气设备的安全控制中起到了至关重要的作用。通过低电流控制高电流,继电器能够有效地隔离控制电路与被控设备,从而保障使用者的安全。本项目将介绍如何通过树莓派Pico与继电器模块结合,使用C++编程实现继电器的控制。 一…...
题解 - 找子序列(2024.12上海月赛丙组T4)
题目描述 Dave 有一个长度为 n 的非负整数序列 a1-n, 和一个非负整数 m 。 他希望知道是否有一个 a 的非空子序列,使得子序列中所有元素的按位与(bitwise AND)结果为 m。 换言之,他想知道是否存在一个下标序列 i1-k(k ≥ 1),满足 1 ≤ i1 < i2 < …...
在centos 7.9上面安装mingw交叉编译工具
1.说明 为了在centos上面编译windows的程序,需要安装mingw工具,mingw工具是可以编译windows程序的一些工具链,使用方式和linux一致 2.下载脚本 使用脚本方式编译,github的脚本位置:https://github.com/Zeranoe/ming…...
ubuntu wine mobaxterm找不到串口和解决方案
安装好 打开MobaXterm时发现,没有可供选择的串口 我们再检查wine设备映射 ls -la ~/.wine/dosdevices/ 串口是存在的,我们再来一番神操作,并没有回滚操作,不知是否是必要修改 打开 注册表,在HKEY_LOCAL_MACHINE中的…...
如何编译安装系统settings设置应用(5.0.0-Release)
本文介绍如何在OpenHarmony 5.0.0 r版本中修改系统设置应用,并且编译安装到开发板上 开发环境 1.dayu200开发板 2.OpenHarmony 5.0.0r 固件 3.API12 full sdk (如果安装full sdk过程中出现报错hvigor ERROR: Cannot find module typescript,请参考 h…...
<项目代码>YOLOv8 车牌识别<目标检测>
项目代码下载链接 <项目代码>YOLOv8 车牌识别<目标检测>https://download.csdn.net/download/qq_53332949/90121387YOLOv8是一种单阶段(one-stage)检测算法,它将目标检测问题转化为一个回归问题…...
协同办公软件新升级:细节优化,让办公更简单
细节决定成败,企业酷信协同办公系统通过贴近客户实际需求的一系列改进和创新,在技术架构、系统结构、管理理念和使用性能上,都达到了国内先进水平,同时具备独特的优势。让我们看看企业酷信是如何通过这些细节提升,为企…...
【原创学习笔记】西门子1200 PLC实现变频器控制
一、实现的功能及应用的场合 通过PLC的不同指令,发送指令控制电机的启停和速度大小 二、硬件配置 1、西门子1214 PLC 2.TIA V16 3.SINAMICS G120C 三、实现功能步骤 1.添加设备G120C PN-调整以太网地址 根据实际情况选择有无滤波器,电机参数…...
SQL server学习02-使用T-SQL创建数据库
目录 一, 使用T-SQL创建数据库 1,数据库的存储结构 2,创建数据库的语法结构 1)使用T-SQL创建学生成绩管理数据库 二,使用T-SQL修改数据库 1,修改数据库的语法结构 1)修改学生成绩管理数…...
2024153读书笔记|《春烂漫:新平摄影作品选》——跳绳酷似人生路,起落平常,进退平常,莫惧征途万里长
2024153读书笔记|《春烂漫:新平摄影作品选》——跳绳酷似人生路,起落平常,进退平常,莫惧征途万里长 《春烂漫:新平摄影作品选》作者新平,2019.12.25年读完的小书,当时就觉得挺不错,今…...
MySQL有哪些高可用方案?
大家好,我是锋哥。今天分享关于【MySQL有哪些高可用方案?】面试题。希望对大家有帮助; MySQL有哪些高可用方案? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 MySQL 高可用方案旨在确保数据库系统的高可靠性、低宕机时间、以及在硬件故障…...
深入剖析AI大模型:大模型时代的 Prompt 工程全解析
今天聊的内容,我认为是AI开发里面非常重要的内容。它在AI开发里无处不在,当你对 AI 助手说 "用李白的风格写一首关于人工智能的诗",或者让翻译模型 "将这段合同翻译成商务日语" 时,输入的这句话就是 Prompt。…...
Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...
Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
DBAPI如何优雅的获取单条数据
API如何优雅的获取单条数据 案例一 对于查询类API,查询的是单条数据,比如根据主键ID查询用户信息,sql如下: select id, name, age from user where id #{id}API默认返回的数据格式是多条的,如下: {&qu…...
Python如何给视频添加音频和字幕
在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...
爬虫基础学习day2
# 爬虫设计领域 工商:企查查、天眼查短视频:抖音、快手、西瓜 ---> 飞瓜电商:京东、淘宝、聚美优品、亚马逊 ---> 分析店铺经营决策标题、排名航空:抓取所有航空公司价格 ---> 去哪儿自媒体:采集自媒体数据进…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...
[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】,分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...
【Nginx】使用 Nginx+Lua 实现基于 IP 的访问频率限制
使用 NginxLua 实现基于 IP 的访问频率限制 在高并发场景下,限制某个 IP 的访问频率是非常重要的,可以有效防止恶意攻击或错误配置导致的服务宕机。以下是一个详细的实现方案,使用 Nginx 和 Lua 脚本结合 Redis 来实现基于 IP 的访问频率限制…...

