数据结构·复杂度

目录

1 时间复杂度

2 大O渐进表示法

举例子(计算时间复杂度为多少)

3 空间复杂度


前言:复杂度分为时间复杂度和空间复杂度,两者是不同维度的,所以比较不会放在一起比较,但是时间复杂度和空间复杂度是用来衡量一个算法是否好坏的标准,时间复杂度用来描述算法运行的时间快慢,空间复杂度用来衡量一个算法所需要的额外空间。最初的计算机时代计算机的存储量很小,所以额外注重空间复杂度,随着发展,计算机的存储已经不是让人担心的点了,所以更为注重时间复杂度。


1 时间复杂度

时间复杂度的定义上可以认为使劲按复杂度是一个函数,定量的描述了算法所需要的时间,但是理论上来说,运行的时间是要上机测试才能测试出来的,实际测试就会花很多时间,所以有了时间复杂度这个分析方式分析算法中执行的基本操作的次数,认定为时间复杂度。

int main()
{
	for (int i = 0; i < N; i++)
	{
		for (int j = 0; j < N; j++)
		{
			//测试语句
		}
	}
	for (int i = 0; i < N; i++)
	{
		//测试语句
	}
	int M = 10;
	while (M--)
	{
		//测试语句
	}
	return 0;
}

这段代码的时间复杂度是?

两个嵌套的for循环,也就是执行了N^2次,再来一个for循环,执行次数为N,最后while收尾,执行次数为10,所以时间复杂度的函数为F(N) = N^2 + N + 10,随着N的增大,式子的值会大到无法想象 ,这都得益于N^2,那么整个式子的决定性因素是N^2,所以我们就认为时间复杂度是N^2。

这种估算的方法被称为大O渐进表示法,只取式子中的决定性因素。


2 大O渐进表示法

计算时间复杂度的时候我们通常采用大O渐进表示法,推导大O阶的表示方法为:

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

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

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

这里其实完全可以类比高中的数学函数,y = x,x可以是2x也可以是3x,这就是为什么舍弃常数的原因,x和2x是没有区别的。

那么什么是加法常数呢?只要没有循环或者递归,我们都可以认为执行次数为O(1),哪怕是写了一万行代码。

执行程序的时候会存在最好情况 最坏情况 以及平均情况(期望值),一般计算时间复杂度的时候都是计算的最坏情况,所以像冒泡排序,运气好点,时间复杂度就是O(1),运气不好就是O(N^2)了。

举例子(计算时间复杂度为多少)

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

第一个for循环的执行次数为2 * N次,第二个while循环执行次数为10次,那么函数表达式就是2*N + 10,根据大O表示法,时间复杂度为O(N)。

void Fun3(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 ", count);
}

第一个for循环执行次数为M次,第二个for循环的执行次数为N次,那么函数表达式就是O(M + N),

那么最后结果视结果而定,如果M远大于N,那么时间复杂度就是O(M),那么N >> M同理可得。

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

根据第一个for循环的循环条件,循环次数为100次,是常数,所以时间复杂度就是O(1)。

void BubbleSort(int* a, int n)
{
	assert(a);
	for (size_t end = n; end > 0; end--)
	{
		int flag = 0;
		for (size_t i = 1; i < end; i++)
		{
			if (a[i - 1] > a[i])
			{
				swap(&a[i - 1], &a[i]);
				flag = 1;
			}
		}
		if (flag == 0)
		{
			break;
		}
	}
}

冒泡排序严格意义上来说不是标准的O(N^2),因为它的执行次数在随着程序的执行是逐渐减少的,最开始两两比较的次数是N- 1,每趟下来,就会确定一个数据的位置,所以两两比较的次数每次都会减去1,那么if这个语句执行的次数就是N - 1 N - 2……1,利用高中的等差数列的知识,可以得到时间复杂度的函数是F(N) = (N - 1) * (N - 1 + 1) / 2,根据大O表示法,可得得出复杂度为O(N^2)。

int BinarySeatch(int* a, int n, int x)
{
	assert(a);
	int begin = 0;
	int end = n - 1;
	//[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个值,就有N/2/2/2/2/2/2……=  1,那么执行次数就是找的次数就是除以2的次数,计算方式就是

N = 2^x,x是执行的次数,我们求的就是x的值,那么利用高中的换底公式,我们可以得到

x是以2为底,N的对数,而因为对数不太好写,除非使用专业的公式编辑器,所以默认规定LogN,表示的就是以2为底,N的对数,如果有其他底数,就照常写。

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

计算阶乘递归的时间复杂度,递归,会多次开辟函数栈帧,所以一次递归,就会开辟一个函数栈帧,执行次数是1,那么递归n次,总执行次数就是N,所以时间复杂度就是O(N)。

引申:

long long Fac(size_t N)
{
	if (0 == N)
		return 1;
	for (int i = 0; i < N; i++)
	{
		//测试语句
	}
	return Fac(N - 1) * N;
}

 递归函数里面加了一个for循环,时间复杂度是多少呢?

不加for循环之前,每个函数的时间复杂度是O(1),开辟了N个函数,那么复杂度就是O(N),但是现在每个函数的时间复杂度就是O(N)了,因为每个子函数里面都有一个for循环,所以N个O(N)的函数放在一起,时间复杂度就是O(N ^ 2)。

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

我们知道递归用来计算斐波那契数列是非常不划算的,因为重复计算的次数太多了,是次方级别的重复:

像这样,函数执行的次数是从2的0次方开始,一直到2的N次方,那么利用高中的等比数列的公式,可以得出时间复杂度为函数为F(N) = 2^N - 1,所以时间复杂度就是O(2 ^ N)。

这是非常恐怖的,所以计算斐波那契数列的话还是使用迭代吧!


3 空间复杂度

时间复杂度可以理解为语句的执行次数,空间复杂度可以理解额外开辟的空间,也是采用的大O渐进表示法。当然,空间复杂度不是多少字节,意义不大,因为当前的计算机存储足够大,所以空间复杂度表示的是变量的个数,需要注意的是:函数需要的栈空间在编译期间就已经确定好了,所以计算空间复杂度靠的是程序运行时候显示出来的额外空间。

void BubbleSort(int* a, int n)
{
	assert(a);
	for (size_t end = n; end > 0; end--)
	{
		int flag = 0;
		for (size_t i = 1; i < end; i++)
		{
			if (a[i - 1] > a[i])
			{
				swap(&a[i - 1], &a[i]);
				flag = 1;
			}
		}
		if (flag == 0)
		{
			break;
		}
	}
}

计算冒泡排序的空间复杂度:

 这个函数里面有变量,但是是有限个的,如end flag i ,并没有额外开辟空间,所以空间复杂度是O(1),毕竟是没有额外申请空间的。

那么递归计算斐波那契数列的空间复杂度是多少呢?

在此之前我们先看这串代码:

void Test1()
{
	int a = 10;
	printf("%p\n", &a);
}
void Test2()
{
	int b = 10;
	printf("%p\n", &b);
}
int main()
{
	Test1();
	Test2();
	return 0;
}

试问运行结果是不是一样的?有人就会问了,不同函数存的地址肯定不是一样的啊,一般情况是这样的,但是如果两个函数连续调用,并且函数的功能一样,对空间的需求是一样的,那么内存在栈上的空间分配就不会额外开辟空间,也就是说一个函数调用完后,另一个函数就会接着使用这块空间,所以地址是一样的。

那么类比到斐波那契数列的重复计算里面:

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

实际上上开辟的空间都是为了计算函数Fib(N)到Fib(1)的,函数的功能是一样,对空间的需求也是一样的,所以重复计算的时候就不会单独开辟空间,所以需要计算的,都用了开辟的Fib(N)到Fib(1)的空间,那么空间复杂度就是O(N)。


感谢阅读!

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

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

相关文章

LeetCode 每日一题 Day 95-101

2917. 找出数组中的 K-or 值 给你一个整数数组 nums 和一个整数 k 。让我们通过扩展标准的按位或来介绍 K-or 操作。在 K-or 操作中&#xff0c;如果在 nums 中&#xff0c;至少存在 k 个元素的第 i 位值为 1 &#xff0c;那么 K-or 中的第 i 位的值是 1 。 返回 nums 的 K-o…

中医药专家学者齐聚丽江 把脉健康产业新发展

3月10日&#xff0c;中国丽江第二届健康产业发展论坛暨全国卫生健康技术推广传承应用项目技能大赛在丽江&#xff08;国际&#xff09;民族文化交流中心隆重开幕。国内生物科学、中医学领域的多名专家学者齐聚丽江&#xff0c;探讨健康产业的未来发展趋势&#xff0c;为丽江乃至…

C/C++语言学习基础版(一)

目录 一and二、C语言说明 注释&#xff1a; 1、声明语句 2、输出函数 3、return 语句 三、C语言的数据结构 1、常量与变量 2、基本数据结构 3、关键字 练习&#xff1a;进制转换 四、基本输入输出 1、字符输出函数putchar 2、字符输入函数getchar 3、格式化输出函…

在家不无聊,赚钱有门道:5个正规线上赚钱平台,轻松开启副业

随着网络技术的快速发展&#xff0c;越来越多的人开始寻求通过网络来探索兼职副业的可能性&#xff0c;期望实现额外的收入。在这个过程中&#xff0c;选择一个正规且可靠的线上兼职平台显得尤为关键。 为此小编精心网上盘点了5个正规且靠谱的线上兼职副业平台。这些平台不仅安…

C语言深入理解指针(1)

前言 小陈也是学完了指针&#xff0c;还是有很多不多的地方&#xff0c;接下来会输出5篇博客去帮助自己彻底弄懂指针&#xff0c;以前的知识也需要复盘了呀。 内存和地址 1.1 内存 举个例子&#xff0c;去理解这两个的词&#xff0c;一个外卖员去送外卖&#xff0c;他首先需…

Halcon 使用光流算子检测运动物体

文章目录 算子optical_flow_mg 计算两个图像之间的光流vector_field_length 计算向量场的向量长度select_shape_std 选择给定形状的区域vector_field_to_real 将矢量场图像转换为两个实值图像intensity 计算灰度值的均值和偏差local_max_sub_pix 以亚像素精度检测局部极大值 Ha…

手撸nano-gpt

nano GPT 跟着youtube上AndrejKarpathy大佬复现一个简单GPT 1.数据集准备 很小的莎士比亚数据集 wget https://raw.githubusercontent.com/karpathy/char-rnn/master/data/tinyshakespeare/input.txt 1.1简单的tokenize 数据和等下的模型较简单&#xff0c;所以这里用了个…

飞塔防火墙开局百篇——002.FortiGate上网配置——在路由模式下使用虚拟接口对(virtual-wire-pair)

在路由模式下使用虚拟接口对&#xff08;virtual-wire-pair&#xff09; 拓扑配置接口配置策略 使用方有透明模式下一进一出的这样需求的组网&#xff0c;可以在路由模式下使用虚拟接口对&#xff08;virtual-wire-pair&#xff09;替代。 登陆FortiGate防火墙界面&#xff0c;…

城市基础信息管理系统 (VB版电子地图源码/公交车线路图/超市平面图)-143-(代码+程序说明)

转载地址http://www.3q2008.com/soft/search.asp?keyword143 请访问 以下地址,查看最新版本, 新增加支持 建筑物 距离测量, 鸟瞰, 地图放大缩小, VB完善地图扩充程序(城市街道基础信息管理系统 )-362-&#xff08;代码&#xff0b;&#xff09; 这套系统印象深刻 因为,写了一…

12双体系Java学习之局部变量和作用域

局部变量 局部变量的作用域 参数变量

数据结构中的堆(Java)

文章目录 把普通数组转换大顶堆数组堆增删改查替换堆排序 把普通数组转换大顶堆数组 该方式适用索引为0起点的堆 在堆&#xff08;Heap&#xff09;这种数据结构中&#xff0c;节点被分为两类&#xff1a;叶子节点&#xff08;Leaf Nodes&#xff09;和非叶子节点&#xff08;N…

面试旺季,鸿蒙开发岗位怎么能没有面试题刷呢?

一年一度的面试浪潮来袭&#xff0c;你是否也想着利用这次机会去实现&#xff0c;跳槽涨薪的梦呢&#xff1f;在往年这个时候基本就有许多的小伙伴跑找到我要相关的面试题进行刷题&#xff0c;或要简历模板对自己的简历进行优化。 今年我又整了点新鲜的面试题&#xff0c;如果…

Linux系统之ipcalc命令的基本使用

Linux系统之ipcalc命令的基本使用 一、ipcalc命令介绍二、ipcalc命令的使用帮助2.1 ipcalc命令的help帮助信息2.2 ipcalc命令的语法解释 三、ipcalc命令的基本使用3.1 计算子网掩码3.2 计算网络地址3.3 找出所对应的主机名3.4 计算子网详细信息 四、ipcalc命令使用注意事项 一、…

由于 Positive Technologies 的专业知识,Moxa 消除了工业无线转换器中的一个漏洞。

我们的专家在 NPort W2150A 和 W2250A 转换器中发现了该漏洞 - 这些设备可将工业控制器、仪表和传感器连接到本地 Wi-Fi 网络。Moxa 已根据负责任的披露政策通知了该威胁&#xff0c;并发布了软件更新。 &#x1f977; 攻击者可以完全访问这些设备。 Positive Technologies 公…

目标检测应用场景—数据集【NO.28】无人机红外目标检测数据集

写在前面&#xff1a;数据集对应应用场景&#xff0c;不同的应用场景有不同的检测难点以及对应改进方法&#xff0c;本系列整理汇总领域内的数据集&#xff0c;方便大家下载数据集&#xff0c;若无法下载可关注后私信领取。关注免费领取整理好的数据集资料&#xff01;今天分享…

(二)运行自己的stable-diffusion

前面的步骤如https://datawhaler.feishu.cn/docx/BwjzdQPJRonFh8xeiSOcRUI3n8b所示 拷贝、解压文件后&#xff0c;进入到stable-diffusion-webui的文件夹中&#xff0c;文件如下&#xff1a; 启动&#xff1a; 运行效果&#xff1a; 由于生成了好几个图&#xff0c;所以…

为什么不要使用elasticsearch

互联网上有很多文章&#xff0c;都在讲为什么要使用elasticsearch&#xff0c;却很少有人讲为什么不要使用elasticsearch。作为深入研究elasticsearch四年&#xff0c;负责公司万亿级别检索的操盘手&#xff0c;借着这篇文章&#xff0c;给大家分享一下&#xff0c;为什么不要使…

nginx swrr负载均衡算法的二宗罪及其改进的思考

目录 1. swrr负载均衡算法的二宗罪1.1 第一宗罪: 共振引起系统崩溃1.2 第二宗罪: 吃CPU大户 2. 对swrr负载均衡算法的改进的思考2.1 “共振”问题的解决2.2 “吃CPU大户”问题的解决 1. swrr负载均衡算法的二宗罪 swrr是一种基于加权轮询的负载均衡算法。它根据服务器的权重来分…

一款 Windows C盘文件清理工具

推荐一款 Windows C盘清理工具 0. 引言1. 下载地址 0. 引言 Windows 在使用过程&#xff0c;C盘的空间会变得越来越少。 Windows在C盘放了很多缓存&#xff0c;临时文件&#xff0c;我们自己还不敢乱删。 今天试了1款工具&#xff0c;可以很方便的查看C盘各个文件夹的文件大小…

中间件 | RabbitMq - [AMQP 模型]

INDEX 1 全局示意2 依赖 1 全局示意 AMQP&#xff0c;即高级消息队列协议&#xff08;Advanced Message Queuing Protocol&#xff09;&#xff0c;整体架构如下图 producer 发送消息给 rabbit mq brokerrabbit mq broker 分发消息给 consumer消费producer/consumer 都通过 …