数据结构·复杂度

news2025/1/13 3:18:48

目录

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)。


感谢阅读!

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

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

相关文章

Ubuntu 安装腾讯会议

1.官网下载 进入腾讯会议下载官网下载腾讯会议Linux客户端 选择x86_64格式安装包下载 若不知道自己的系统架构,输入 uname -a 在命令行结果中查看系统架构信息 2.终端命令安装 cd {你的下载路径} sudo dpkg -i TencentMeeting_0300000000_3.19.0.401_x86_64_default.publi…

数据结构·二叉树(一)

1. 树概念及结构 1.1 树的概念 树是一种非线性的数据结构&#xff0c;它是由n(n>0)个有限节点组成一个具有层次关系的集合。把它叫做树是因为它看起来像一颗倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;树叶朝下的。 有一个特殊的节点&#xff0c;称为根节点&#…

嵌入式系统工程师错题总结

笔者来介绍一下嵌入式系统工程师考试的一些易错题目 题目介绍  流水线指令计算公式&#xff1a;一条指令总时间max&#xff08;单个指令执行时间&#xff09;*&#xff08;指令数-1&#xff09;  平均故障间隔时间  ICMP协议&#xff1a;传送通信问题相关的消息。 …

sqlserver中将csv非空间数据(带点坐标)转为空间数据

1、导入csv数据 2、修改字段shape为空间字段 ALTER TABLE FJPOIHB66 ALTER COLUMN shape geometry;3、空间字段转字符串 UPDATE FJPOIHB66 SET shape geometry::STGeomFromText(CONVERT(nvarchar(254),shape), 4326);4、设置主键字段 5、即可

141 Linux 系统编程18,线程,ps –Lf 进程 查看LWP,线程间共享数据,优缺点,编译加-lpthread,

一 线程概念 什么是线程 LWP&#xff1a;light weight process 轻量级的进程&#xff0c;本质仍是进程(在Linux环境下) 进程&#xff1a;独立地址空间&#xff0c;拥有PCB 线程&#xff1a;有独立的PCB&#xff0c;但没有独立的地址空间(共享) 区别&#xff1a;在于是否共…

C++vector的模拟实现

文章目录 模拟实现1. 迭代器2. 容量操作&#xff08;1&#xff09;size和capacity&#xff08;2&#xff09;reserve&#xff08;3&#xff09; resize 3. 元素访问&#xff08;1&#xff09;下标 [ ] 4. 修改操作&#xff08;1&#xff09;push_back&#xff08;2&#xff09…

【死磕Elasticsearch】从实战中来,到实战中去

文章目录 写在前面&#xff1a;1、索引阻塞的种类2、什么时候使用阻塞&#xff1f;场景1&#xff1a;进行系统维护场景。场景2&#xff1a;保护数据不被随意更改场景。场景3&#xff1a;优化资源使用的场景。场景4&#xff1a;遵守安全规则场景。 3、添加索引阻塞API4、解除设置…

使用jquery的autocomplete属性实现联想补全操作

平时使用百度&#xff0c;淘宝等软件搜索时&#xff0c;常见一个搜索框联想提示&#xff0c;感觉确实好用但没有研究过原理&#xff0c;最近恰巧工作中遇到一个同样的场景&#xff0c;不同于大厂使用高端的Python&#xff0c;这次需要使用jQuery的autocomplete属性来自动联想补…

git撤回代码提交commit或者修改commit提交注释

执行commit后&#xff0c;还没执行push时&#xff0c;想要撤销之前的提交commit 撤销提交 使用命令&#xff1a; git reset --soft HEAD^命令详解&#xff1a; HEAD^ 表示上一个版本&#xff0c;即上一次的commit&#xff0c;也可以写成HEAD~1 如果进行两次的commit&#xf…

Singularity(二)| 安装singularity工具

Singularity&#xff08;二&#xff09;| 安装singularity工具 以默认安装 Ubuntu 22.04 (jammy) 发行版的 WSL 2 (Windows Subsystem for Linux 2) 为例&#xff1a; 参考&#xff1a;官方快速安装向导 安装系统依赖项 首先在主机上安装开发工具和库&#xff0c;在基于 debian…

【黑马程序员】python函数

文章目录 函数什么是函数为什么学习函数函数定义函数的传入参数函数的返回值返回值基础None返回值 函数说明文档函数的嵌套调用定义代码示例 全局变量和局部变量全局变量global变量局部变量 函数综合案例 函数 什么是函数 组织好的&#xff0c;可重复使用的、用来实现特定功能…

5.Java并发编程—JUC线程池架构

JUC线程池架构 在Java开发中&#xff0c;线程的创建和销毁对系统性能有一定的开销&#xff0c;需要JVM和操作系统的配合完成大量的工作。 JVM对线程的创建和销毁&#xff1a; 线程的创建需要JVM分配内存、初始化线程栈和线程上下文等资源&#xff0c;这些操作会带来一定的时间和…

【C语言步行梯】自定义函数、函数递归详谈

&#x1f3af;每日努力一点点&#xff0c;技术进步看得见 &#x1f3e0;专栏介绍&#xff1a;【C语言步行梯】专栏用于介绍C语言相关内容&#xff0c;每篇文章将通过图片代码片段网络相关题目的方式编写&#xff0c;欢迎订阅~~ 文章目录 什么是函数库函数自定义函数函数执行示例…

数据结构 第2章:线性表

文章目录 2.1 线性表的定义和操作2.1.1 线性表的基本概念2.1.2 线性表的基本操作 2.2. 顺序表2.2.1. 顺序表的基本概念2.2.2. 顺序表的实现2.2.3. 顺序表的基本操作 2.3 链表2.3.1 单链表的基本概念2.3.2 单链表的实现2.3.3 单链表的插入2.3.4. 单链表的删除2.3.5. 单链表的查找…

VB 数据质量诊断软件(分析数据的完整性,合理性,准确性)-139-(代码+程序说明)

转载地址http://www.3q2008.com/soft/search.asp?keyword139 前言: 为何口出狂言,作任何VB和ASP的系统, 这个就是很好的一个证明 :) 又有些狂了... 数据库操作谁都会,接触的多了也没什么难的,VB编程难在哪?算法上,这个是一个算法题的毕业设计 哈哈忙活了足足有一○小时, …

2024年最新阿里云和腾讯云云服务器价格租用对比

2024年阿里云服务器和腾讯云服务器价格战已经打响&#xff0c;阿里云服务器优惠61元一年起&#xff0c;腾讯云服务器61元一年&#xff0c;2核2G3M、2核4G、4核8G、4核16G、8核16G、16核32G、16核64G等配置价格对比&#xff0c;阿腾云atengyun.com整理阿里云和腾讯云服务器详细配…

C语言 ——关键字

关键字&#xff1a;在C语言中被赋予了特定含义的英文单词&#xff0c;一共有32个关键字 * 关键字全部小写 * 在特定的编译器中&#xff0c;关键字是高亮显示的 vs&#xff1a;蓝色或者紫色 vs&#xff1a;蓝色 下图圈起来的都是关键字 c auto break case char const con…

Kafka消费者重平衡

「&#xff08;重平衡&#xff09;Rebalance本质上是一种协议&#xff0c;规定了一个Consumer Group下的所有Consumer如何达成一致&#xff0c;来分配订阅Topic的每个分区」。 比如某个Group下有20个Consumer实例&#xff0c;它订阅了一个具有100个分区的Topic。 正常情况下&…

【C语言】如何规避野指针

✨✨ 欢迎大家来到莉莉的博文✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 目录 一、概念&#xff1a; 二、野指针成因&#xff1a; 2.1. 指针未初始化 2.2 指针越界访问 3. 指针指向的空间释放 三、如何规避野指针 3.…

MT6771 android13 自定义背光曲线

一. Android系统源码中的参数配置 MTK6771平台MTK重写了背光曲线的参数&#xff0c;路径在s0_vnd/vendor/mediatek/proprietary/packages/overlay/vendor/FrameworkResOverlayExt/brightness_adaptive_support/res/values/config.xml 不过MTK的其他平台可能不是在这个路径 来看…