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

news2024/9/29 11:28:01

目录

1.什么是算法?

1.1算法的复杂度

2.算法的时间复杂度

2.1 时间复杂度的概念

计算Func1中++count语句总共执行了多少次

2.2 大O的渐进表示法

2.3常见时间复杂度计算举例 

实例1:执行2N+10次

实例2:执行M+N次

实例3:执行了100000000次

实例4:计算strchr的时间复杂度

实例5:计算BubbleSort的时间复杂度

实例6:计算BinarySearch的时间复杂度

实例7: 计算阶乘递归Fac的时间复杂度

实例8:计算斐波那契递归Fib的时间复杂度

3.算法的空间复杂度

实例1:计算BubbleSort的空间复杂度

实例2:计算Fibonacci的空间复杂度

实例3:计算阶乘递归Fac的空间复杂度

4.常见复杂度对比


1.什么是算法?

算法:

算法 (Algorithm):就是定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为 输出。简单来说算法就是一系列的计算步骤,用来将输入数据转化成输出结果。
常见应用于排序/ 二分查找
算法特点:

1.有穷性。一个算法应包含有限的操作步骤,而不能是无限的。事实上“有穷性”往往指“在合理的范围之内”。如果让计算机执行一个历时1000年才结束的算法,这虽然是有穷的,但超过了合理的限度,人们不把他视为有效算法。

2. 确定性。算法中的每一个步骤都应当是确定的,而不应当是含糊的、模棱两可的。算法中的每一个步骤应当不致被解释成不同的含义,而应是十分明确的。也就是说,算法的含义应当是唯一的,而不应当产生“歧义性”。

3. 有零个或多个输入、所谓输入是指在执行算法是需要从外界取得必要的信息。

4. 有一个或多个输出。算法的目的是为了求解,没有输出的算法是没有意义的

5.有效性。 算法中的每一个 步骤都应当能有效的执行。并得到确定的结果。

1.1算法的复杂度

算法在编写成可执行程序后,运行时需要耗费时间资源和空间 ( 内存 ) 资源 。因此 衡量一个算法的好坏,一般是从时间空间两个维度来衡量的 ,即时间复杂度和空间复杂度。
时间复杂度主要衡量一个算法的 运行快慢 ,而空间复杂度主要衡量一个算法运行 所需要的额外空间(需要多少内存) 。在计算 机发展的早期,计算机的存储容量很小。所以对空间复杂度很是在乎。但是经过计算机行业的迅速发展,计算机的存储容量已经达到了很高的程度。所以我们如今已经不需要再特别关注一个算法的空间复杂度。

2.算法的时间复杂度

2.1 时间复杂度的概念

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

计算Func1中++count语句总共执行了多少次

让我们来实践一下吧:
// 请计算一下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);
}

时间复杂度的函数式(也就是Func1 执行的基本操作次数):

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

但是这个表达式,太准确,太细节,太繁琐了。时间复杂度不是准确地计算出这个数学函数式的执行次数,而是给它分一个级别,它到底是哪个量级的。

举例:就像马云和马化腾,不需要关心他们的账户具体几分几毛,只需要知道他们是富豪就行了。
准确值F(N)=N^2+2N+10估算值O(N^2)
N = 10
F(N) = 130 100
N = 100
F(N) = 10210
10000
N = 1000
F(N) = 1002010
1000000
结论1:
N越大,后面项对结果影响越小,也就是说 阶数最高(N^2)的那一项就是影响最大的,保留最高阶项。
结论2:
通过上面我们会发现大O的渐进表示法去掉了那些对结果影响不大的项,简洁明了的表示出了执行次数。
实际中我们计算时间复杂度时,我们其实并不一定要计算精确的执行次数,而只需要 大概执行次数,那么这 里我们使用大 O 的渐进表示法。
只要用大O这个东西来表示就说明它是一个估算的值。

2.2 O的渐进表示法

O 符号( Big O notation ):是用于描述函数渐进行为的数学符号。
推导大 O 阶方法:
1 、用常数 1 取代运行时间中的所有加法 常数
2 、在修改后的运行次数函数中, 只保留最高阶项
3 、如果最高阶项存在且不是 1 ,则 去除 与这个项目 相乘的常数 。得到的结果就是大 O 阶。
有些算法的时间复杂度存在最好、平均和最坏情况:
最坏情况:任意输入规模的最大运行次数 ( 上界 )
平均情况:任意输入规模的期望运行次数
最好情况:任意输入规模的最小运行次数 ( 下界 )
例如:在一个长度为 N 数组中搜索一个数据 x:
最好情况: 1 次找到
最坏情况: N 次找到
平均情况: N/2 次找到
在实际中一般情况关注的是算法的 最坏运行情况 ,所以数组中搜索数据时间复杂度为 O(N)

2.3常见时间复杂度计算举例 

实例1:执行2N+10次

// 计算Func2的时间复杂度?
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);
}
基本操作执行了2N+10次,通过推导大O阶方法知道,时间复杂度为 O(N)

实例2:执行M+N次

// 计算Func3的时间复杂度?
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);
}
实例 2 基本操作执行了 M+N 次,有两个未知数 M N ,时间复杂度为 O(N+M)
不能说N无限大了,M就不重要了。除非说会给出一个关系:
N远大于M,则时间复杂度为O(N)
M远大于N,则时间复杂度为O(M)
M等于N或二者相差不大时,则时间复杂度为O(M+N)

实例3:执行了100000000次

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

int main()
{
	Func4(100000);
	Func4(1);

	return 0;
}

执行:实际上cpu的速度是非常快的,相差的执行次数可以忽略,所以时间复杂度依旧为O(1)

O(1)不是代表1次,而是代表常数次,就算k<10亿,它也是O(1)

我们平时能写到的常数最大也就是40多亿左右(整型能表示的范围),cpu是可以承受的。

实例3基本操作执行了100000000次,通过推导大O阶方法,时间复杂度为 O(1)

实例4:计算strchr的时间复杂度

// 计算strchr的时间复杂度?
const char * strchr ( const char * str, int character );

这是关于strchr的模拟实现

#include<stdio.h>
#include<assert.h>
char* my_strchr(const char* str, const char ch)
{
	assert(str);
	const char* dest = str;
	while (dest != '\0' && *dest != ch)
	{
		dest++;
	}
	if (*dest == ch)
		return (char*)dest;
	return NULL;
}
int main()
{
	char* ret = my_strchr("hello", 'l');
	if (ret == NULL)
		printf("不存在");
	else
		printf("%s\n", ret);
	return 0;
}

我的理解就是strchr和strstr的区别:就是strstr是输入一个字符串在主串中查找,而strchr是输入一个字符,然后在主串中查找。这个链接有关于strstr的知识点:http://t.csdn.cn/NEaip

若查找失败,返回NULL。查找成功则返回首字符的地址,然后打印的时候一直到'\0'结束

所以说:

指明了这个数组的长度然后去查找它的时间复杂度才是O(1),长度不明确的话,长度就是N,那么需要递归N次,时间复杂度就是O(N)
实例 4 基本操作执行最好 1 次,最坏 N 次,时间复杂度一般看最坏,时间复杂度为 O(N)

实例5:计算BubbleSort的时间复杂度

// 计算BubbleSort的时间复杂度?
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;
 }
}
图解:

 那么比较的次数构成等差数列:用等差数列求和公式得到最后的执行次数是F(N)=(N-1)*N/2;

这题关于循环的不是说有两层循环嵌套就直接判断它的时间复杂度是O(N^2),因为如果比较次数是已知的(外层循环n<10,内层循环n<1000000)那就是O(1) ,而且冒泡排序会有优化版本,在有序的情况下,他的时间复杂度是O(N),只走外层循环。
实例 5 基本操作执行最好 N 次,最坏执行了 (N*(N+1)/2 次,通过推导大 O 阶方法 + 时间复杂度一般看最坏,时间复杂度为 O(N^2)

实例6:计算BinarySearch的时间复杂度

// 计算BinarySearch的时间复杂度?
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;

}

图解:

假设找了x次,那么除了x个2

2^x =N  --> x = log2N   

所以说可以从最后一次查找一直乘2,乘到原数组的长度

实例6基本操作执行最好1次,最坏O(log2N)次,时间复杂度为 O(log2N) ps:logN在算法分析中表示是底数为2,对数为N。有些地方会写成lgN。

实例7: 计算阶乘递归Fac的时间复杂度

计算下面两段代码的时间复杂度
//实例7:
// 计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{
	if (0 == N)
		return 1;

	return Fac(N - 1) * N;
}

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

	for (size_t i = 0; i < N; i++)
	{

	}

	return Fac(N - 1) * N;
}

图解:

左边的每一次函数调用里面的for循环语句(如果有其它循环语句也会算上)的执行次数,左边的1表示它是常数次,而不是1次(就说如果函数里面没什么循环语句,有几个if语句,那时间复杂度也是O(1))。

右边的简单来说就说有N+1个函数调用,而每一个函数调用里面的循环语句都执行了N+1次,所以应该把每次的递归调用的函数里面的循环语句都加起来。

补充的点:

时间是加起来的,不是乘起来的,就比如说上图的return Fac(N-1)*N:表示的是上一次的结果乘N,但是执行次数也是一次,因为这个地方的*对于计算机来说仅仅是一个指令。时间复杂度算的是这个程序在走的过程中这个指令的执行次数。

实例8:计算斐波那契递归Fib的时间复杂度

// 计算斐波那契递归Fib的时间复杂度?
long long Fib(size_t N)
{
	if (N < 3)
		return 1;

	return Fib(N - 1) + Fib(N - 2);

}

图解:

 执行次数之和符合等比数列:使用错位相减法

这题可以这样理解:

关于那个三角形,白色的区域在N越大的情况下,就会远远大于黑色区域,而时间复杂度是用来计算大致计算某个数学函数是的量级的,给它分一个级别,所以可以看作是满项的状态下计算,然后执行次数之和构成等比数列,用大O渐进表示法去算时间复杂度为O(2^N)。

3.算法的空间复杂度

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

实例1:计算BubbleSort的空间复杂度

// 计算BubbleSort的空间复杂度?
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(1)

而关于形参int *a,和int n,它们不会被算在空间复杂度中。

实例2:计算Fibonacci的空间复杂度

// 计算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;
 fibArray[1] = 1;
 for (int i = 2; i <= n ; ++i)
 {
 fibArray[i] = fibArray[i - 1] + fibArray [i - 2];
 }
 return fibArray;
}

空间复杂度,它计算的是你在这个函数内部开辟了多少额外空间,如果是常数个的话,就是O1,如果开辟的大小不确定,一般就是O(N)。

所以说空间复杂度为O(N)。

实例3:计算阶乘递归Fac的空间复杂度

// 计算阶乘递归Fac的空间复杂度?
long long Fac(size_t N)
{
 if(N == 0)
 return 1;
 
 return Fac(N-1)*N;
}

图解:

递归调用了N层每次调用建立一个栈帧,每个栈帧使用了常数个空间O(1)
由于这里调用了N个函数,同时没有返回,所以合起来就是O(N)

实例4:计算斐波那契递归Fib的空间复杂度(两个递归)

// 计算斐波那契递归Fib的空间复杂度?
long long Fib(size_t N)
{
 if(N < 3)
 return 1;
 
 return Fib(N-1) + Fib(N-2);
}

前言:

空间的销毁不是整没了那块空间,是归还使用权,归还给操作系统

因为内存空间是属于操作系统进程的,比如说让你malloc一块空间,就获得这块空间的使用权,free一下就把空间使用权还给操作系统了

时间是一去不复返时间是累积计算的,空间是可以重复利用不累积计算
简单的说,右边的函数和左边的函数共用一个栈帧。

代码运行:栈是向下生长的,调用Func1和Func2相当于共用一块空间,因为Func1销毁之后,到Func2创建,位置还是那个位置,地址也是那个地址。

因为主函数的a和Func1的a在不同栈帧里面,所以可以同名。

实例4递归调用了N次,开辟了N个栈帧,每个栈帧使用了常数个空间。空间复杂度为O(N)

调用时,建立栈帧;

返回时,销毁。(归还给操作系统)

4.常见复杂度对比

一般算法常见的复杂度如下:

图解:

 本章完,如有不足之处,欢迎大佬指正。

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

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

相关文章

java jwt生成token并在网关设置全局过滤器进行token的校验并在给请求头设置参数及在微服务中解析参数

1、首先引入jjwt的依赖 <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt</artifactId><version>0.9.1</version> </dependency>2、编写生成token的工具类 package com.jjw.result.util;import com.jjw.res…

软考高级系统架构设计师(九) 作文模板-论设计模式及其应用(未完待续)

目录 掌握的知识点 创建型 结构型 行为型 掌握的知识点 设计模式分为哪3类 每一类包含哪些具体的设计模式 创建型 创建型模式是对对象实例化过程的抽象&#xff0c;他通过抽象类所定义的接口&#xff0c;封装了系统中对象如何创建、组合等信息。 创建型模式主要用于创建对…

【物联网】微信小程序接入阿里云物联网平台

微信小程序接入阿里云物联网平台 一 阿里云平台端 1.登录阿里云 阿里云物联网平台 点击进入公共实例&#xff0c;之前没有的点进去申请 2.点击产品&#xff0c;创建产品 3.产品名称自定义&#xff0c;按项目选择类型&#xff0c;节点类型选择之恋设备&#xff0c;联网方式W…

Linux下安装Redis的详细安装步骤

一.Redis安装 1.下载linux压缩包 【redis-5.0.5.tar.gz】 2.通过FlashFXP把压缩包传送到服务器 3.解压缩 tar -zxvf redis-5.0.5.tar.gz4.进入redis-5.0.5可以看到redis的配置文件redis.conf 5.基本的环境安装 使用gcc -v 命令查看gcc版本已经是4.8.5了&#xff0c;于是就…

ubuntu系统突然失去网络问题

修复ubuntu系统网络问题 1. 服务不存在&#xff1f;2. 修改配置&#xff0c;自动启动网络 每天都在用的ubuntu系统突然ssh连接不上&#xff0c;进系统ifconfig也不显示ip。当然也ping不通任何网页。 1. 服务不存在&#xff1f; 初步怀疑网络服务被关闭了&#xff0c;需要修改配…

【C6】数据类型/移植/对齐,内核中断,通过IO内存访问外设,PCI

文章目录 1.内核基础数据类型/移植性/数据对齐&#xff1a;页大小为PAGE_SIZE&#xff0c;不要假设4K&#xff0c;保证可移植性1.1 kdatasize.c&#xff1a;不同的架构&#xff08;x86_64,arm&#xff09;&#xff0c;基础类型大小可能不同&#xff0c;主要区别在long和指针1.2…

chatgpt赋能python:用Python访问数据库的SEO文章

用Python访问数据库的SEO文章 在当今互联网飞速发展的时代&#xff0c;数据处理和数据库技术的重要性不言而喻。在这些应用中&#xff0c;Python是使用最广泛和最受欢迎的编程语言之一。Python的简单和易学性使其成为理想的选项&#xff0c;可以通过Python来访问各种类型的数据…

荣耀90推出最新MagicOS7.1更新,增加控制中心功能

荣耀 90 系列机型推出了最新的 Magic OS 7.1更新&#xff0c;版本号为7.1.0.137 (C00E130R2P2)。该更新主要增加了控制中心功能&#xff0c;并对部分场景拍摄效果进行了优化。此外&#xff0c;该更新还提升了系统与部分三方应用的兼容性&#xff0c;以提高系统性能和稳定性。 …

选择最适合您自动化系统的控制方式

自动化系统可采用多种不同的控制方式&#xff0c;其中硬件控制和PLC&#xff08;可编程逻辑控制器&#xff09;是常见的选择。 刚好&#xff0c;我这里有上位机入门&#xff0c;学习线路图&#xff0c;各种项目&#xff0c;需要留个6。 硬件控制通常指使用专用硬件电路实现控…

C++3(sizeof和逗号运算符,类型转换)

1.sizeof的用法 逗号运算符 口诀&#xff1a;从左到右算&#xff0c;返回最右边的值 类型转换 如何实现的隐式类型转换&#xff1f; 先算右边的&#xff0c;右边的3&#xff08;int&#xff09;先提升为double &#xff0c;然后算得&#xff08;7.541&#xff08;double&#…

CMU 15-445 -- 关系型数据库重点概念回顾 - 01

CMU 15-445 -- 关系型数据库重点概念回顾 - 01 引言Relational Data ModelDBMS数据模型Relational ModelRelation & TuplePrimary KeysForeign Keys Data Manipulation Languages (DML)Relational Algebra Advanced SQLSQL 的历史SQLAggregatesGroup ByHavingOutput Redire…

内存屏障类型表

load store 啥意思 内存屏障类型表 StoreLoad Barriers是一个“全能型”的屏障&#xff0c;它同时具有其他3个屏障的效果。现代的多处理器大多支持该屏障&#xff08;其他类型的屏障不一定被所有处理器支持&#xff09;。执行该屏障开销会很昂贵&#xff0c;因为当前处理器通常…

在文件每行开头或结尾插入指定字符

1、在文件每行插入指定字符 sed -i "s/^/curl /g" missing.txt效果 2、在每行末尾插入指定字符 sed -i "s/$/结束 /g" missing.txt

leetcode1856. 子数组最小乘积的最大值(单调栈-java)

子数组最小乘积的最大值 leetcode1856.子数组最小乘积的最大值题目描述解题思路代码演示&#xff1a; 经典算法集锦 leetcode1856.子数组最小乘积的最大值 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/maximum-subarr…

【网络安全】初探SQL注入漏洞

如何利用SQL注入漏洞获取用户密码 前言1. 设计思路2. 设计目的 一、网站快速搭建1. 登录页2. 注册页3. 数据库连接页4. 首页&#xff08;登录后跳转到此处&#xff09;5. session页6. 注销页7. 查询页8. 数据库 二、SQL注入实例&#xff08;小试牛刀&#xff09;1. 猜测漏洞逻辑…

tomcat部署以及优化

目录 1.三个核心组件 2.tomcat服务部署 3.虚拟主机配置 4.tomcat优化 5.部署两台tomcat服务器 6.总结 1.三个核心组件 web容器 完成web服务 servlet容器 名为catalina 用于处理servlet JSP容器 将JSP动态网页翻译成…

网络通信之旅:揭秘交换机、路由器与网关的神奇世界!

文章目录 一 交换机2.1 交换机初识2.2 MAC地址表2.3 数据包2.4 交换机与数据包2.5 泛洪2.6 结论&#xff1a;交换机—二层设备 三 路由器3.1 WAN口&LAN口3.2 路由器-WAN交换机 四 网关4.1 子网划分4.2 网关4.3 路由 五 实践&#xff1a;路由器桥接-搭建主副路由器5.1 知识探…

动态规划:

这类问题非常简单&#xff0c;甚至看起来有点笨&#xff0c;说白了就是利用计算机的计算能力一步步算过去&#xff0c;也就是大多数人没有意识到的递推问题 比如求1~n的前缀和&#xff1a; #include<iostream> using namespace std; long long sum[100]; int main(){in…

20kV高精度可调高压稳压测试电源的学习与使用

一&#xff1a;应用范围 A: 二极管反向耐压测试 B: 二极管反向漏电流测试 C: 高压电容耐压测试 D: 玻璃釉电阻非线性性能测试 E:氙灯击穿电压测试 F: 材料耐压测试 二、特点 高精度恒流恒压高压输出源 它拥有0~20kV的电压输出能力, 0.005%的电压分辨率精度, 0.1uA的电 …

Docker安装Prometheus和Grafana监控Redis

Docker安装Prometheus和Grafana监控Redis 使用 Docker 安装 Grafana 和 Prometheus 无疑是最简单的&#xff0c;我们接下来将采用此种方式。 1、安装Prometheus 查看Prometheus镜像 $ docker search prometheus拉取镜像 $ docker search bitnami/prometheus在/home/zhangs…