数据结构--算法的时间复杂度和空间复杂度

文章目录

  • 算法效率
  • 时间复杂度
    • 时间复杂度的概念
    • 大O的渐进表示法
    • 计算实例
  • 时间复杂度
    • 实例
  • 常见复杂度对比
  • 例题

算法效率

算法效率是指算法在计算机上运行时所消耗的时间和资源。这是衡量算法执行速度和资源利用情况的重要指标。

例子:

long long Fib(int N)
{
	if(N < 3)
		return 1;
	return Fib(N-1) + Fib(N-2);
}

这是一个斐波那契函数,用的是递归的计算方法,每次创建函数就会在栈区开辟一块空间,递归次数越多,开辟空间越多;
所以,代码的简洁说明不了算法的效率

算法效率的评估主要包括时间复杂度和空间复杂度:

  1. 时间复杂度:时间复杂度描述了算法执行所需的时间随输入规模增加而增长的趋势。常见的时间复杂度包括常数时间O(1)、线性时间O(n)、对数时间O(log n)、平方时间O(n²)等。通过分析算法中关键操作的执行次数来确定时间复杂度,通常使用大O符号表示。

  2. 空间复杂度:空间复杂度描述了算法在执行过程中所需的额外存储空间随输入规模增加而增长的趋势。常见的空间复杂度包括常数空间O(1)、线性空间O(n)、对数空间O(log n)等。通过分析算法中使用的数据结构和辅助空间来确定空间复杂度。

评估算法效率时,我们希望选择具有更低时间复杂度和空间复杂度的算法,以提高程序的执行速度和资源利用率。但需要注意的是,时间复杂度和空间复杂度通常存在着一定的取舍关系,有时需要在时间和空间之间做出权衡。

除了时间复杂度和空间复杂度,还可以考虑一些实际情况下的算法效率问题,如最坏情况、平均情况和最好情况下的执行时间,以及算法在特定硬件环境下的性能等。综合考虑这些因素可以更全面地评估算法的效率。

为了提高算法效率,我们可以采用一些常见的优化方法,如减少循环次数、使用合适的数据结构和算法、剪枝和缓存等。同时,也可以借助工具和框架来提升算法效率,如并行计算、GPU加速、分布式计算等。

时间复杂度

时间复杂度的概念

在计算机科学中,算法的时间复杂度是一个函数,它定量描述了该算法的运行时间。一个算法执行所耗费的时间,从理论上说,是不能算出来的,只有你把你的程序放在机器上跑起来,才能知道。但是我们需要每个算法都上机测试吗?是可以都上机测试,但是这很麻烦,所以才有了时间复杂度这个分析方式。一个算法所花费的时间与其中语句的执行次数成正比例,算法中的基本操作的执行次数,为算法的时间复杂度。

例子:计算Func1中的++count语句执行了多少次?

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

两个循环嵌套就是N^2,再一个单独的循环2*N,加上10;
那么

F(N)=N^2+2*N+10;

N=10时,F(N)=130
N=100时,F(N)=10210
N=1000时,F(N)=1002010
会发现N越大,F(N)越接近N^2;
在实际计算时间复杂度中,我们其实不一定要精准的计算出执行的次数,只需要大概的执行次数,那么这里我们使用大O的渐进表示法;

大O的渐进表示法

大O符号(Big O notation):是用于描述函数渐进行为的数学符号。是一种用于衡量算法时间复杂度的渐进表示方法;

推导大O阶方法
1、用常数1取代运行时间中的所有加法常数。
2、在修改后的运行次数函数中,只保留最高阶项。
3、如果最高阶项存在且不是1,则去除与这个项目相乘的常数。得到的结果就是大O阶。

用大O的渐进表示法,Func1的时间复杂度为:O(N^2)

N=10时,F(N)=100
N=100时,F(N)=10000
N=1000时,F(N)=1000000
通过计算我们发现,这样的表示方法去掉那些对结果影响不大的项,简明了洁表示出执行次数;
对于一些算法,会有好坏的情况,对于这种情况我们取最坏的情况
例如:在一个长度为N数组中搜索一个数据x
最好情况:1次找到
最坏情况:N次找到
平均情况:N/2次找到
时间复杂度O(N);

计算实例

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

在这里M是一个常数项,N是一个未知数,所以我们可以先把常数项略去;然后就只剩下2N,通过计算规则,省去最高项的常数;那么这个函数的时间复杂度为O(N);

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

在这里,M与N都是未知数,且它们是同阶的,所以对于这种情况就要分类讨论:
M与N差不多大,那么时间复杂度:O(M)或O(N)
M远大于N,时间复杂度:O(M)
N远大于M,时间复杂度:O(N)

void Func4(int N)
{
	int count = 0;
	for (int k = 0; k < 100; ++ k)
	{
		++count;
	}
	printf("%d\n", count);
}

这里给出了明确的次数,所以时间复杂度为O(1);

const char * strchr ( const char * str, int character );

查找字符串中第一个出现的字符返回指向 C 字符串
str 中第一个出现的字符的指针。
终止空字符被视为 C 字符串的一部分。因此,也可以定位它以检索指向字符串末尾的指针。

这个就是在字符串中寻找一个字符,对于这种就要分好坏情况;
最好1次,最坏N次,时间复杂度一般看最坏,时间复杂度为 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;
	}
}

这是一个冒泡排序,很明显这个两个循环在嵌套;外循环执行1次,内循环就得执行N次;那么外循环执行N次,总共就执行N^2次
时间复杂度O(N^2);

int BinarySearch(int* a, int n, int x)
{
	assert(a);
	int begin = 0;
	int end = n-1;
	// [begin, end]:begin和end是左闭右闭区间,因此有=号
	while (begin <= end)
	{
		int mid = begin + ((end-begin)>>1);
		if (a[mid] < x)
			begin = mid+1;
		else if (a[mid] > x)
			end = mid-1;
		else
			return mid;
	}
	return -1;
}

这是一个二分查找函数,也叫折半查找;我们以最坏的情况去看,每循环一次这个数组的长度就会减半,假设以N表示数组的长度,我们要在数组中寻找一个数,那么假设寻找了x次,那么通过计算2^x=N;再通过换算就是x=log2 N;2是对数的底数,由于底数不好表示,所以对于这个函数的时间复杂度就是为O(logN);

在这里插入图片描述

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

这是一个阶乘递归函数,每递归一次N减1,直至为0,才返回;
那么时间复杂度为O(N)

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

这是开头的例子,斐波那契递归函数,每次在函数中就会递归到两个函数中去;
在这里插入图片描述
递归到最后,会发现像个金字塔一样,全部加起来F(N)=1+2+4+8+……+2(N-1),由数学的等比求和公式得F(N)=2^N-1;
那么时间复杂度为F(N)=2^N;

时间复杂度

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

实例

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)

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

这里用了malloc函数在堆区开辟了N个额外空间,所以
空间复杂度为O(N)

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

这里用了递归的方法,每递归一次就会在栈帧中开辟一次空间,总共开辟N次,每次空间内为常数次大,所以
空间复杂度O(N);

常见复杂度对比

在这里插入图片描述

在这里插入图片描述

例题

在这里插入图片描述

在这里插入图片描述
这里用三种办法来解答:
第一种
在这里插入图片描述

时间复杂度:O(N^2)
空间复杂度:O(1)

这里利用这种方法,就是用两个循环嵌套,最终就得出了复杂度;

第二种
在这里插入图片描述
通过开辟额外空间,将对应位置的数字进行放入新数组中,最后再返回到原数组中;

时间复杂度:O(N)
空间复杂度:O(N)

void rotate(int* nums, int numsSize, int k){
	//开辟额外空间
	int* new=(int*)malloc(sizeof(int)*numsSize);
	//当k长度超过数组长度时
	k%=numsSize;
	memcpy(new,nums+numsSize-k,sizeof(int)*k);
	memcpy(new,nums,sizeof(int)*(numsSize-k));
	memcpy(nums,new,sizeof(int)*numsSize);

	free(new);
	
}

第三种
在这里插入图片描述
利用倒置再倒置的方法就将数组右旋;

时间复杂度:O(N)
空间复杂度:O(1)

//将数组进行倒置
void reserve(int* sem,int left,int right)
{
    while(left<right)
    {
        int tmp=sem[left];
        sem[left]=sem[right];
        sem[right]=tmp;
        left++;
        right--;
    }

}
void rotate(int* nums, int numsSize, int k){
    
    k%=numsSize;//如果k的长度大于numsize,需要取余
    //利用部分数组倒置,在全数组倒置的方法进行轮转
    reserve(nums,0,numsSize-k-1);
    reserve(nums,numsSize-k,numsSize-1);
    reserve(nums,0,numsSize-1);

    
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

利用Python 实现 模拟退火算法

模拟退火算法 模拟退火算法流程图 初始温度 新温度值 进入循环 生成新的解 按照 bound按照 constraint 计算新解与当前解的目标差异判断是否接受解判断是否更新解循环结束按照温度降低率降低温度温度小于最低温度输出结果 模拟退火算法示例代码1 import numpy as npdef objec…

如何用arduino uno主板播放自己想要的曲子。《我爱你中国》单片机版本。

目录 一.效果展示 二.基本原理 三.电路图 四.代码 一.效果展示 arduino播放《我爱你中国》 二.基本原理 利用arduino uno单片机实现对蜂鸣器振动频率的调节&#xff0c;基于PWM控制系统通过代码实现控制。 三.电路图 四.代码 //main.uno #define Buzzer 2int PotBuffer …

vue启动失败问题

解决办法情况1&#xff1a;确认自己是否进入了vuedemo项目的目录。 解决办法情况2&#xff1a;目录进入正确npm start错误&#xff0c;这时可以进入自己电脑的项目文件中去删除node_modules和package-lock.json&#xff0c;然后回到控制台npm i或npm install安装依赖&#xff0…

【MIPI协议 D-PHY基础介绍】

MIPI协议 D-PHY基础介绍 前言一、MIPI介绍1.1 D-PHY MIPI 简单介绍1.2 C-PHY MIPI 简单介绍1.3 M-PHY MIPI 简单介绍 二、D-PHY具体介绍2.1 DSI分层结构2.2 D-PHY电气特性介绍2.3 D-PHY 工作模 三、LINE线上的模式3.1 line线上 state code3.2 high speed data line传输3.3 Low …

9.环境对象和回调函数

9.1环境对象 指的是函数内部特殊的变量this&#xff0c;它代表着当前函数运行时所处的环境 作用&#xff1a; 弄清楚this的指向&#xff0c;可以让我们代码更简洁 ➢函数的调用方式不同&#xff0c;this指代的对象也不同 ➢[谁调用&#xff0c;this 就指代谁] 是判断this指向的…

记一次 .NET 某物流API系统 CPU爆高分析

一&#xff1a;背景 1. 讲故事 前段时间有位朋友找到我&#xff0c;说他程序CPU直接被打满了&#xff0c;让我帮忙看下怎么回事&#xff0c;截图如下&#xff1a; 看了下是两个相同的程序&#xff0c;既然被打满了那就抓一个 dump 看看到底咋回事。 二&#xff1a;为什么会打…

ElasticSearch学习之ElasticSearch快速入门实战

1.先“分词” 2.倒排索引&#xff08;前提是分词&#xff09; ElasticSearch官网地址&#xff1a;欢迎来到 Elastic — Elasticsearch 和 Kibana 的开发者 | Elastichttps://www.elastic.co/cn/ 一、下载 下载地址&#xff1a;https://www.elastic.co/cn/downloads/past-re…

记一次 HTTPS 抓包分析和 SNI 的思考

日常听说 HTTPS 是加密协议&#xff0c;那现实中的 HTTPS 流量&#xff0c;是真的完全加密吗&#xff1f; ——答案是&#xff0c;不一定。原因嘛&#xff0c;抓个包就知道了。 我们用 curl 命令触发一下&#xff1a; curl -v https://s-api.37.com.cn/api/xxx * Trying 1…

Redis篇

文章目录 Redis-使用场景1、缓存穿透2、缓存击穿3、缓存雪崩4、双写一致5、Redis持久化6、数据过期策略7、数据淘汰策略 Redis-分布式锁1、redis分布式锁&#xff0c;是如何实现的&#xff1f;2、redisson实现的分布式锁执行流程3、redisson实现的分布式锁-可重入4、redisson实…

【已解决】windows7添加打印机报错:加载Tcp Mib库时的错误,无法加载标准TCP/IP端口的向导页

windows7 添加打印机的时候&#xff0c;输入完打印机的IP地址后&#xff0c;点击下一步&#xff0c;报错&#xff1a; 加载Tcp Mib库时的错误&#xff0c;无法加载标准TCP/IP端口的向导页 解决办法&#xff1a; 复制以下的代码到新建文本文档.txt中&#xff0c;然后修改文本文…

java 企业工程管理系统软件源码+Spring Cloud + Spring Boot +二次开发+ MybatisPlus + Redis

&#xfeff; 电子招标采购软件 解决方案 招标面向的对象为供应商库中所有符合招标要求的供应商&#xff0c;当库中的供应商有一定积累的时候&#xff0c;会节省大量引入新供应商的时间。系统自动从供应商库中筛选符合招标要求的供应商&#xff0c;改变以往邀标的业务模式。招…

【ROS 02】ROS通信机制

机器人是一种高度复杂的系统性实现&#xff0c;在机器人上可能集成各种传感器(雷达、摄像头、GPS...)以及运动控制实现&#xff0c;为了解耦合&#xff0c;在ROS中每一个功能点都是一个单独的进程&#xff0c;每一个进程都是独立运行的。更确切的讲&#xff0c;ROS是进程&#…

滑动窗口例题

1.适合解决的题目类型 滑动窗口&#xff0c;可以用来解决一些查找满足一定条件的连续区间的性质&#xff08;长度等&#xff09;的问题。由于区间连续&#xff0c;因此当区间发生变化时&#xff0c;可以通过旧有的计算结果对搜索空间进行剪枝&#xff0c;这样便减少了重复计算…

造船厂船只维修人员定位系统:提高修船效率和安全性

引言&#xff1a;造船厂是一个复杂而危险的工业环境&#xff0c;船只维修过程需要高效的协作和精确的定位。为了提高修船效率和安全性&#xff0c;造船厂船只维修人员定位系统应运而生。 本文华安联大将介绍该系统的功能和作用&#xff0c;以及如何通过定位技术和智能分析来实…

提高电脑寿命的维护技巧与方法分享

在维护电脑运行方面&#xff0c;我有一些自己觉得非常有用的技巧和方法。下面我将分享一些我常用的维护技巧&#xff0c;并解释为什么我会选择这样做以及这样做的好处。 首先&#xff0c;我经常清理我的电脑内部的灰尘。电脑内部的灰尘会影响散热效果&#xff0c;导致电脑发热…

搞懂异地多活,看这篇就够了

目录 01 系统可用性 02 单机架构 03 主从副本 04 风险不可控 05 同城灾备 06 同城双活 07 两地三中心 08 伪异地双活 09 真正的异地双活 10 如何实施异地双活 11 异地多活 总结 后记 在软件开发领域&#xff0c;「异地多活」是分布式系统架构设计的一座高峰&#…

Resnet与Pytorch花图像分类

1、介绍 1.1数据集介绍 flower_data├── train│ └── 1-102&#xff08;102个文件夹&#xff09;│ └── XXX.jpg&#xff08;每个文件夹含若干张图像&#xff09;├── valid│ └── 1-102&#xff08;102个文件夹&#xff09;└── ─── └── XXX.jp…

html5播放器视频切换和连续播放的实例

当前播放器实例可以使用changeVid接口切换正在播放的视频。当有多个视频&#xff0c;在上一个视频播放完毕时&#xff0c;自动播放下一个视频时也可采用该处理方式。 const option {vid: 88083abbf5bcf1356e05d39666be527a_8,//autoplay: true,//playsafe: , //PC端播放加密视…

Maven右侧依赖Dependencies消失

项目右侧的Maven依赖Dependencies突然消失&#xff0c;项目中的注解都出现报错&#xff0c;出现这种情况应该是因为IDEA版本早于maven版本&#xff0c;重新检查项目中的Maven路径&#xff0c;选择File->Settings->搜索Maven&#xff0c;检查Maven home directory&#xf…

SHELL——备份脚本

编写脚本&#xff0c;使用mysqldump实现分库分表备份。 1、获取分库备份的库名列表 [rootweb01 scripts]# mysql -uroot -p123456 -e "show databases;" | egrep -v "Database|information_schema|mysql|performance_schema|sys" mysql: [Warning] Using …