数据结构(1)——算法时间复杂度与空间复杂度

news2025/2/8 3:50:39

目录

前言

一、算法

1.1算法是什么?

1.2算法的特性

1.有穷性

2.确定性

3.可行性

4.输入

5.输出

二、算法效率

2.1衡量算法效率

1、事后统计方法

2、事前分析估计方法

2.2算法的复杂度

2.3时间复杂度

2.3.1定义

2.3.2大O渐进表示法

2.3.3常见时间复杂度举例

1、O(N)

2、O(N+M)

3.O(1)

4、O(N²)

5、O(logN)

6、O(N)递归

7、O(N²)斐波那契

2.4空间复杂度

2.4.1定义

2.4.2空间复杂度举例

1、O(1)

2、O(N)

2.5常见的函数增长率

总结


前言

“数据结构”是计算机程序设计的重要理论技术基础,它不仅是计算机的核心课程,而且已成为其它理工专业的热门选修课。                                                     ——《数据结构C语言版》严蔚敏


一、算法

1.1算法是什么?

首先,我们总能听见算法算法的,到最后一问你算法是什么?你支支吾吾的回答说不就是一些计算的方式吗?难不成还有其它解释方法。对此,在严蔚敏的书中是这么解释的:

算法(algorithm)是对特定问题求解步骤的一种描述,它是指令的有限序列,其中每一条指令表示一个或者多个操作

我们生活中的问题都需要一定步骤的解决,算法亦如此,你解决一个问题,需要有先后的顺序和方法,最后才能解决好,经过这些操作和步骤,整合起来才是这特定问题的算法。

1.2算法的特性

当然解决问题的算法也是有一些特性的,在这里说明出来:

1.有穷性

一个算法必须是又穷的,要不然在解决什么问题,我们要的是通过算法来获取最后的结果,实现最后的结果,而不是无穷下去,最后什么都得不到。当然是对一些合法的输入值,每一步都可以在有穷的时间内完成,最后也在有穷步之后结束。

2.确定性

算法中每一条操作和指令必须要有明确的含义,这样才能确定要做什么,要干什么,在任何条件下,算法只有唯一的一条执行路径,即对于相同的输入只能得出相同的输出。你得到最后这个结果后,你不能再来一次后不得这个结果了,那么肯定就是出问题了。

3.可行性

这个算法必须是可行的,因为你不可行那你在算什么,算法中的实现都是可以通过已经实现的基本运算执行有限的次数下实现的。

4.输入

一个算法有零个或者多个的输入,这些输入取自某个特定的对象的集合。像有一些不用输入就只需要进行操作的命令。

5.输出

一个算法有一个或者多个的输出,这些输出是同输入有着某些特定的关系的量。

而且通常一个算法的好坏,看看有没有下面的五条:

1、正确性

2、可读性

3、健壮性

4、效率与低存储量需求

如果你的算法满足了这些,那么它就是一个“好”的算法。

二、算法效率

2.1衡量算法效率

一般衡量算法效率有两种方法:

1、事后统计方法

因为很多计算机内部都有计时器的功能,有些可能会精细到毫秒级别,不同的算法的程序可以通过一组或者成千上万组的数据来区分这个算法的优势和劣势,但有这种方式有一个缺陷,就是每一次衡量都需要先运行依据算法编制的程序,并且所得的时间的统计量还依赖于计算机硬件,软件环境等因素,所以有时候会掩盖算法本身的优势和劣势。

2、事前分析估计方法

一个高级程序语言编写的程序在计算机上消耗的时间取决于算法选用的策略、问题的规模、书写程序的语言(因为语言级别越高,执行效率就越低)、编译程序所产生的机器代码的质量、机器执行指令的速度。基于这些要求,我们可以用一个问题规模的函数来分析估计。

2.2算法的复杂度

我们之前提到了算法满足五条就是一个“”的算法,但其中的第四条提到了“效率与低存储需求”,这里判断的方法就涉及到了时间与空间两个方面,也就是看时间复杂度和空间复杂度。

时间复杂度主要衡量的是一个算法的运行速度的快慢,而空间复杂度主要是衡量一个算法运行所需要的额外空间。

2.3时间复杂度

2.3.1定义

在计算机科学中,算法的时间复杂度是一个函数,它定量描述了该算法的运行时间,一个算法执行所耗费的时间,从理论上来说,是不能算出来的,但一个算法所花费的时间与其中的语句的执行次数成正比例,算法中的基本操作的执行次数,为算法的时间复杂度。

例如下面的代码:

for (int i = 0; i < N; i++)
{
	for (int j = 0; j < N; j++)
	{
		sum=i+j
	}
}

我们只需要计算基本语句和问题规模N的数学表达式就可以。

这里是两次循环,每一个循环都是N次循环,所以F(N)=N²。而在这里我们用大O表示法来表示时间复杂度,也就是O(N²)。

2.3.2大O渐进表示法

实际我们在计算机时间复杂度的时候,并不需要一定要计算精准准确的执行次数,而只需要大概执行次数就可以。

大O符号:用于描述函数渐进行为的数字符号

推导大O阶的方法:

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

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

3、如果高阶项存在且不是1,则去除与这个项目相乘的常数,得不到的结果就是大O阶。

有些算法的时间复杂度会存在最好、平均和最坏的情况:

最坏情况:任意输入规模的最大运行次数(上界)

平均情况:任意输入规模的期望所运行次数

最好情况:任意输入规模的最小运行次数(下界)

例如我们如果在一组数中找一个数:

1 2 3 4 5 6 7 8 9....N

如果找1的话就是最好的情况一次就找到了,最坏的情况就是N次找到,平均就是N/2次找到。

在实际中一般情况关注的是算法最坏的情况,所以它的时间复杂度就是O(N)。

2.3.3常见时间复杂度举例

1、O(N)

void Func1(int N)
{
	int cout = 0;
	for (int n = 0; n < 2 * N; n++)
	{
		++cout;
	}
	int cout1 = 5;
	while (cout1--)
	{
		++cout;
	}
	printf("%d \n", cout);
}

这里我们可以进行推导,来计算基本语句和问题规模N的函数等于什么?这里经历了一个循环N次,接下来又是一个常数的循环,最后也就是F(N)=N+5。

因为时间复杂度里面有常数要舍去,所以最后用大O表示也就是O(N)。

2、O(N+M)

void Func2(int N, int M)
{
	int cout = 0;
	for (int i = 0; i < N; i++)
	{
		cout++;
	}
	for (int i = 0; i < M; i++)
	{
		cout++;
	}
	printf("%d \n", cout);
}

这里同样的,计算基本语句和问题规模N,M的数学表达式:

第一个循环循环了N次,第二个循环了M次,所以最后是F(N,M)=N+M,时间复杂度也就是O(N+M)。

3.O(1)

void Func3(int N)
{
	int cout = 0;
	for (int i = 0; i < 10000; i++)
	{
		cout++;
	}
	printf("%d \n", cout);
}

这里也就是找N和基本语句的表达式,我们发现这里就一个循环了10000次,是一个常数,用大O表示就是O(1)。

4、O(N²)

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;
	}
}

这里是一个简单的冒泡排序,Swap是用来交换数字的,这里计算一下问题规模n的函数表达式;

这是一个嵌套循环,外面一层循环走了n次,里面的从1开始循环,如果前一个大于当前的数字就交换,这里可以发现得需要除以2,所以最好的情况下就是一趟循环就找到了,也就是O(N),最坏就是两层,也就是F(N)=N*(N+1)/2,也就是O(N²)。

5、O(logN)

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;
}

这是一个经典的用二分查找x的值,这里就是找出问题规模n的表达式。我们知道二分查找是折半的,每一次都是平方倍的,所以n=N²,则F(N)=logN,(以二为敌N的对数),最好情况就是O(1),最坏就是O(logN)。

6、O(N)递归

long long Fac4(size_t N)
{
	if (0 == N)
		return 1;
	return Fac4(N - 1) * N;
}

这里给出了一个递归计算阶乘,找问题规模N的表达式,这里我们反着推也就是N*(N-1)*(N-2)...*1,我们发现中间经历的N次,所以F(N)=N;所以用大O表示就是O(N)。

7、O(2ⁿ)斐波那契

long long Fib(size_t N)
{
    if (N <= 1) {
        return N;
    }
    return Fib(N-1) + Fib(N-2);
}


这里就是一个斐波那契数列,这里也是通过递归进行两次两次的实现,所以最后也就是表示N需要2ⁿ次,也就是用大O表示法就是O(2ⁿ)。

2.4空间复杂度

2.4.1定义

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

空间复杂度不是程序占用了多少字节的空间,而是算的是变量的个数,计算规则基本和时间复杂度类似,也是用大O渐进表示法。

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

这里说一下:根据经验大多数空间复杂度都是O(1)或者O(N)。

2.4.2空间复杂度举例

1、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;
	}
}

依旧是冒泡函数,这里使用了常数个变量空间,所以最后空间复杂度就是O(1)。

2、O(N)

long long Fac(size_t N)
{
    if(N == 0)
        return 1;
    return Fac(N-1)*N;
}

这里是那个阶乘的递归,我们知道递归调用的时候是调用自身,并且也是占用栈帧的,所以这里就是调用了N次,也就是开辟了N个栈帧,而每一个栈帧用了常数个变量,所以这里的空间复杂度就是O(N)。

2.5常见的函数增长率

这里给出常见的函数增长率:


总结

今天主要对算法,算法的时间、空间复杂度进行了分析和学习,这里会计算会看就可以。

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

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

相关文章

uniapp小程序自定义中间凸起样式底部tabbar

我自己写的自定义的tabbar效果图 废话少说咱们直接上代码&#xff0c;一步一步来 第一步&#xff1a; 找到根目录下的 pages.json 文件&#xff0c;在 tabBar 中把 custom 设置为 true&#xff0c;默认值是 false。list 中设置自定义的相关信息&#xff0c; pagePath&#x…

C++编程语言:抽象机制:模板(Bjarne Stroustrup)

目录 23.1 引言和概观(Introduction and Overview) 23.2 一个简单的字符串模板(A Simple String Template) 23.2.1 模板的定义(Defining a Template) 23.2.2 模板实例化(Template Instantiation) 23.3 类型检查(Type Checking) 23.3.1 类型等价(Type Equivalence) …

DeepSeek-VL2论文解读:用于高级多模态理解的专家混合视觉语言模型

github:https://github.com/deepseek-ai/DeepSeek-VL2 paper: https://github.com/deepseek-ai/DeepSeek-VL2/blob/main/DeepSeek_VL2_paper.pdf 大型视觉语言模型&#xff08;VLMs&#xff09;已经成为人工智能领域的变革性力量&#xff0c;将大型语言模型&#xff08;LLMs&…

第二个Qt开发实例:在Qt中利用GPIO子系统和sysfs伪文件系统实现按钮(Push Button)点击控制GPIO口(效果为LED2灯的灭和亮)

引言 本文承接博文 https://blog.csdn.net/wenhao_ir/article/details/145420998 里的代码&#xff0c;在那里面代码的基础上添加上利用sysfs伪文件系统实现按钮(Push Button)点击控制GPIO口的代码&#xff0c;进而实现LED2灯的灭和亮。 最终的效果是点击下面的LED按钮实现LED…

Day37-【13003】短文,串的基本概念,匹配算法,算法时间复杂度,真题训练

文章目录 第二节 串串的基本概念串的模式匹配朴素的模式匹配算法(BF算法)算法最坏时间复杂度O(n x m) 改进的模式匹配算法(KMP算法)特征向量next&#xff0c;来确定k值特征向量next的算法实现 算法最坏时间复杂度O(n)进一步改进next值的计算&#xff0c;简化步骤 第四章真题真题…

陷入闭包:理解 React 状态管理中的怪癖

TLDR 闭包就像函数随身携带的背包&#xff0c;包含它们创建时的数据React 组件使用闭包来记住它们的状态和属性过时的闭包可能导致状态更新不如预期时的错误函数式更新提供了一个可靠的方式来处理最新状态 简介 你是否曾经疑惑过&#xff0c;为什么有时你的 React 状态更新不…

【SRC排名】安全应急响应中心SRC上榜记录

2023年 新氧第三 https://security.soyoung.com/top 合合第四 https://security.intsig.com/index.php?m&chall&aindex 2024年 好未来第一 https://src.100tal.com/index.php?m&chall&aindex&#xff08;官网是总榜&#xff0c;年榜只有海报&#xff09;…

Linux——基础命令1

$&#xff1a;普通用户 #&#xff1a;超级用户 cd 切换目录 cd 目录 &#xff08;进入目录&#xff09; cd ../ &#xff08;返回上一级目录&#xff09; cd ~ &#xff08;切换到当前用户的家目录&#xff09; cd - &#xff08;返回上次目录&#xff09; pwd 输出当前目录…

OSPF基础(1):工作过程、状态机、更新

OSPF基础 1、技术背景&#xff08;与RIP密不可分&#xff0c;因为RIP中存在的问题&#xff09; RIP中存在最大跳数为15的限制&#xff0c;不能适应大规模组网周期性发送全部路由信息&#xff0c;占用大量的带宽资源以路由收敛速度慢以跳数作为度量值存在路由环路可能性每隔30秒…

【目标检测】模型验证:K-Fold 交叉验证

K-Fold 交叉验证 1、引言1.1 K 折交叉验证概述 2、配置2.1 数据集2.2 安装包 3、 实战3.1 生成物体检测数据集的特征向量3.2 K 折数据集拆分3.3 保存记录3.4 使用 K 折数据分割训练YOLO 4、总结 1、引言 我们将利用YOLO 检测格式和关键的Python 库&#xff08;如 sklearn、pan…

Unity 2D实战小游戏开发跳跳鸟 - 计分逻辑开发

上文对障碍物的碰撞逻辑进行了开发,接下来就是进行跳跳鸟成功穿越过障碍物进行计分的逻辑开发,同时将对应的分数以UI的形式显示告诉玩家。 计分逻辑 在跳跳鸟通过障碍物的一瞬间就进行一次计分,计分后会同步更新分数的UI显示来告知玩家当前获得的分数。 首先我们创建一个用…

京准:NTP卫星时钟服务器对于DeepSeek安全的重要性

京准&#xff1a;NTP卫星时钟服务器对于DeepSeek安全的重要性 京准&#xff1a;NTP卫星时钟服务器对于DeepSeek安全的重要性 在网络安全领域&#xff0c;分布式拒绝服务&#xff08;DDoS&#xff09;攻击一直是企业和网络服务商面临的重大威胁之一。随着攻击技术的不断演化…

Android学习20 -- 手搓App2(Gradle)

1 前言 昨天写了一个完全手搓的&#xff1a;Android学习19 -- 手搓App-CSDN博客 后面谷歌说不要用aapt&#xff0c;d8这些来搞。其实不想弄Gradle的&#xff0c;不过想着既然开始了&#xff0c;就多看一些。之前写过一篇Gradle&#xff0c;不过是最简单的编译&#xff0c;不涉…

车型检测7种YOLOV8

车型检测7种YOLOV8&#xff0c;采用YOLOV8NANO训练&#xff0c;得到PT模型&#xff0c;转换成ONNX&#xff0c;然后OPENCV的DNN调用&#xff0c;支持C&#xff0c;python,android开发 车型检测7种YOLOV8

IDEA 中集成 Maven,配置环境、创建以及导入项目

目录 在 IntelliJ IDEA 中集成 Maven 并配置环境 1. 打开 IDEA 设置 2. 定位 Maven 配置选项 3. 配置 Maven 路径 4. 应用配置 创建 Maven 项目 1. 新建项目 2. 选择项目类型 3. 配置项目信息 4. 确认 Maven 设置 5. 完成项目创建 导入 Maven 项目 1. 打开导入窗口…

react关于手搓antd pro面包屑的经验(写的不好请见谅)

我们先上代码&#xff0c;代码里面都有注释&#xff0c;我是单独写了一个组件&#xff0c;方便使用&#xff0c;在其他页面引入就行了 还使用了官方的Breadcrumb组件 import React, { useEffect, useState } from react; import { Breadcrumb, Button } from antd; import { …

[含文档+PPT+源码等]精品大数据项目-Django基于大数据实现的心血管疾病分析系统

大数据项目-Django基于大数据实现的心血管疾病分析系统背景可以从以下几个方面进行阐述&#xff1a; 一、项目背景与意义 1. 心血管疾病现状 心血管疾病是当前全球面临的主要健康挑战之一&#xff0c;其高发病率、高致残率和高死亡率严重威胁着人类的生命健康。根据权威机构…

【Rust自学】19.5. 高级类型

喜欢的话别忘了点赞、收藏加关注哦&#xff08;加关注即可阅读全文&#xff09;&#xff0c;对接下来的教程有兴趣的可以关注专栏。谢谢喵&#xff01;(&#xff65;ω&#xff65;) 19.5.1.使用newtype模式实现类型安全和抽象 在 19.2. 高级trait 中&#xff08;具体来说是…

113,【5】 功防世界 web unseping

进入靶场 代码审计 <?php // 高亮显示当前 PHP 文件的源代码&#xff0c;方便开发者查看代码结构和内容 highlight_file(__FILE__);// 定义一个名为 ease 的类 class ease {// 私有属性 $method&#xff0c;用于存储要调用的方法名private $method;// 私有属性 $args&…

leetCode刷题-图、回溯相关

岛屿数量 class Solution { private:int mi;int mj; public:int numIslands(vector<vector<char>>& grid) {mi grid.size() - 1; // i的范围 0~mimj grid[0].size() - 1; // j的范围 0~mjint landnum 0;bool sea false;do {pair<int, int> res …