数据结构开始——时间复杂度和空间复杂度知识点笔记总结

news2024/12/15 8:08:05

好了,经过了漫长的时间学习c语言语法知识,现在我们到了数据结构的学习。

首先,我们得思考一下

什么是数据结构?

数据结构(Data Structure)是计算机存储、组织数据的方式,指相互之间存在一种或多种特定关系的数据元素的集合。


算法又是什么?

定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为输出。简单来说算法就是一系列的计算步骤,用来将输入数据转化成输出结果
 

算法讲究效率

如何衡量一个算法的好坏
如,斐波那契数列(虽然代码简单,但是它的效率高吗?)

long long Fib(int n)
{
if(n < 3)
{
  return 1;
}

    return Fib(n-1) + Fib(n-2);
}

是先(黑色)一步一步进去下去,算到Fib(1)时,再一步一步返回上去(红色) 

这得花费非常多的时间,效率并不高。

因此,要衡量一个算法的好坏,一般是从时间和空间两个维度来衡量的即时间复杂度,空间复杂度

时间复杂度和空间复杂度

时间复杂度和空间复杂度的区别

作用:时间复杂度主要衡量一个算法的运行快慢,而空间复杂度主要衡量一个算法运行所需要的额外空间(由于目前的空间都很大,基本不用担心空间的)

时间复杂度

时间复杂度:算法的时间复杂度是一个函数
一个算法所花费的时间与其中语句的执行次数成正比例,算法中的基本操作的执行次数,为算法的时间复杂度

也就是:找到某条基本语句与问题规模N之间的数学表达式,就是算出了该算法的时间复杂度。


 

 请计算一下Func1中++count语句总共执行了多少次?
void Func1(int N)
{
int count = 0;
for (int i = 0; i < N ; ++ i)
{
for (int j = 0; j < N ; ++ j)
{                                  这里是N^2
++count;
}
}
for (int k = 0; k < 2 * N ; ++ k)
{
++count;                             这里是2*N
}
int M = 10;
while (M--)
{                                     这里是10次
++count;
}

经过计算我们可以得出:是F(N)=N^2+2*N+10

其实,对于求复杂度,我们一般使用大O的渐进表示法,,现在来介绍下:

大O的渐进表示法

1.大O符号:是用于描述函数渐进行为的数学符号

2.规则方法:

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

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

3)、如果最高阶项存在且不是1,则去除与这个项目相乘的常数

最后得到的结果就是大O阶

eg:N(10000000000000)这应该写成什么呢?

答案:仍然是O(1).

有人会问为啥它这个数那么大了,还是用O(1)表示呢?

现在我们来用形象的例子来解释下:

假如那里有座泰山,我们要铲平,
不同视角:会不同
我们人的视角觉得它非常大,
但是放到太阳系里面,就显得非常小了。

同样,对于编程也是一样的道理。

通过上面我们会发现大O的渐进表示法去掉了那些对结果影响不大的项,简洁明了的表示出了执行次数

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

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

最好情况:任意输入规模的最小运行次数(下界)例如:在一个长度为N数组中搜索一个数据x

最好情况:1次找到最坏情况:N次找到平均情况:N/2次找到

所以,我们实际中一般关注的是最坏的情况。

原因:同样是以一个形象的例子来解释

约会的预期管理
情人节约会可能到时的时间
最好:17:00
平均:18:00
最坏:19:00
那么因为只有30%的机会准时到
所以是不是选择说19:00时最好,把预期拉到最低。

常见时间复杂度计算举例

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

所以,函数是不是就是:F(N)=2*N+10

根据大O法:得时间复杂度就是O(N)

例子2:

// 计算Func3的时间复杂度?
void Func3(int N, int M)
{
int count = 0;
for (int k = 0; k < M; ++ k)
{
++count;                       这里是M次
}
for (int k = 0; k < N ; ++ k)
{                                这里是N次
++count;
}
printf("%d\n", count);
}

函数式:F(N)=M+N

注意:这里的M,N都是未知数,并不是具体数

所以根据大O法:时间复杂度O(M+N)
 

例子3:

// 计算Func4的时间复杂度?
void Func4(int N)
{
int count = 0;
for (int k = 0; k < 100; ++ k)
{                                     这里是100次
++count;
}
printf("%d\n", count);
}

函数式:F(N)=100

根据大O法:由于它是常数嘛,所以用1代替

所以时间复杂度:O(1).

例子4:

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

对于指针,如果没有具体说,一般默认都是O(N)

例子5:

void BubbleSort(int* a, int n)
{
assert(a);
for (size_t end = n; end > 0; --end)
 {                                        一般没说的话,默认N次
   int exchange = 0;
   for (size_t i = 1; i < end; ++i)          因为两个for循环,所以是N^2次
   {
     if (a[i-1] > a[i])
     {
       Swap(&a[i-1], &a[i]);
       exchange = 1;
     }
   }
  if (exchange == 0)              因为这里有了判断,它可以改变最好的情况
  break;                          如果没有这句,
 }                                最好也是O(N^2)
                                  因为它无法知道它已经排序了  
 }                                
 

排序

最好(已经排序好大小):O(1)  X有人说是看一眼就知道它的大小排序了,但是你一亿的时候呢,你并不知道其他数字2与这个的大小?

 你是不是也是要一个一个对比下去呢?
所以  时间复杂度: O(N)  √   

最坏:O(N^2)
其实这里不敢数循环,因为以后会学更复杂的
N-1 N-2 N-3....3 2 1  等差数列 算出O(N^2)

例子5:我们之前写过的二分查找

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

 

          四    第三次             第二次                     第一次 

 我们可以发现,这是不是一次查找,我们就可以筛选掉一半。

最坏情况:区间缩放到一个值时,要么找到,要么找不到
假设N是数组个数,x是最坏查找次数
N/2/2/2/2.../2=1
折半查找多少次就除多少个2
假设是x次

2^x=N
x=log N 

这里我们先来说明一下:

接着,我们来看一下二分查找的显著优势:

对比:

N         1000    100W    10亿                  2^10=1024

O(N)      1000    100W   10亿                 2^20=1024*1024

O(log N)  10       20         30                     2^30=1024*1024*1024  

我们可以看到这二分查找极大改变它的时间效率。

但是一个很不友好的一点是,它仅仅局限于排序好的数,所以我们以后也不太实用

 例子6:

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

 这是不是相当于有N+1次,O(N+1)

根据大O法:时间复杂度:O(N)

Fac(N)
Fac(N-1)
Fac(N-2)
....
Fac(1)    
Fac(0)       

例子七:

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

 等比数列 -缺失(常数)

时间:O(2^N)
空间:O(N)

对此,我们可以得出来:

时间一去不复返,不可重复利用
空间用了以后要归还,可以重复利用。

空间复杂度

1.空间复杂度也是一个数学表达式,是对一个算法在运行过程中临时占用存储空间大小的量度
2.空间复杂度算的是变量的个数(而不是程序占用了多少bytes的空间,因为这个也没太大意义)

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

3.空间复杂度的使用规则也是:大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;
   } 
}

我们上面说了空间复杂度算的是变量的个数

我们看到了上面用了三个变量:end,exchange,i。 -->O(3)

根据大O法:O(1).

例子2:绝大多数空间复杂度都是O(1)或O(N)

 计算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;                        为什么要多加1,因为数组从0开始,
   fibArray[1] = 1; 
   for (int i = 2; i <= n ; ++i)
   {
      fibArray[i] = fibArray[i - 1] + fibArray [i - 2];
   }
   return fibArray;
}

上面创建了n+1个空间

所以根据大O法:O(N)

例子3:

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

 值得注意的是

递归调用了N次,开辟了N个栈帧,每个栈帧使用了常数个空间

函数在栈的开辟函数栈帧时是单独开辟的。调用本身时,会再2开辟一块空间作为新的函数栈帧,但是在此外并没有创建额外的变量,所以会认为创建的每一块栈帧空间复杂度变量是常数个。

举个形象例子:

空间销毁是把那个使用权还给操作系统了,并不是说这块空间就没了
内存的申请,就相当于 住酒店一样,当这房间不用了,这不是说把它炸

所以参数从N到0,共递归了N+1次,

根据大O法:O(N)

例子6:

同样,斐波那契数列也是一样。

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

 空间销毁是把那个使用权还给操作系统了,我们上面说过空间是可以重复使用的,

调用某个函数结束后,再次调用此函数,所用的空间与上一次的函数空间是同一个。

所以,同样是递归了N-1次

由大O法:O(N)。

大O的常见的变化表

祝福语:

最后的最后,希望我们都能战胜困难,一步一步地坚持下去,为以后拿到好offer!

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

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

相关文章

【游戏设计原理】7 - 加德纳的多元智能理论

虽然多元智能理论是对认知方式的分类&#xff0c;但它也可以为游戏设计提供丰富的思路和策略&#xff0c;帮助设计师创建更具吸引力、包容性和多样性的游戏。通过理解不同玩家的认知方式和优势&#xff0c;我们可以更精准地设计游戏的元素和玩法&#xff0c;使其能够吸引广泛的…

【OpenCV】基于分水岭算法的图像分割

介绍 分水岭算法&#xff08;Watershed Algorithm&#xff09;是一种基于形态学的图像分割方法&#xff0c;它模仿了地理学中的分水岭概念。在图像处理中&#xff0c;分水岭算法通过模拟水流从山顶流向谷底的过程来分割图像&#xff0c;其中局部极小值点被视为“山谷”&#x…

ubuntu20.04+ROS Noetic 安装PX4+Mavros

文章目录 系统环境安装依赖PX4 安装老版本安装测试环境变量添加版本查看 安装MAVROS&#xff08;二进制安装非源码安装&#xff09;测试 OGC 地面站安装测试mavros与sitl通信参考 系统环境 ubuntu 20.04 ROS Noetic 如果系统安装了Anaconda等虚拟环境管理器&#xff0c;要退出…

Vue Web开发(七)

1. echarts介绍 echarts官方文档 首先我们先完成每个页面的路由&#xff0c;之前已经有home页面和user页面&#xff0c;缺少mail页面和其它选项下的page1和page2页面。在view文件夹下新建mail文件夹&#xff0c;新建index.vue&#xff0c;填充user页面的内容即可。在view下新建…

WordPress插件 Download-block-plugin下载按钮图标美化

WordPress插件 Download-block-plugin下载按钮图标美化

KALI容器虚拟化Docker安装

为什么需要DOCKER 环境一致性保障 开发与生产环境统一&#xff1a;在软件开发过程中&#xff0c;开发环境和生产环境的差异常常导致应用程序出现问题。例如&#xff0c;开发人员在自己的机器上开发了一个 Web 应用&#xff0c;使用了特定版本的操作系统、数据库和编程语言运行…

【LC】240. 搜索二维矩阵 II

题目描述&#xff1a; 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&#xff1a; 每行的元素从左到右升序排列。每列的元素从上到下升序排列。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,4,7,11,15],[2,5,8,12,19],[3,6,…

前端html,vue使用第三方地图详细教程,以百度地图为例,实现地图标注,导航,定位,路线规划,坐标转换

目录 示例&#xff1a; 准备&#xff1a; ?编辑 开始&#xff1a; 1、新建页面&#xff0c;在script标签中引入百度地图的api数据&#xff0c;把自己在控制台创建的应用的ak替换上去 2、创建一个dom对象&#xff0c;设置宽高 3、在js中初始化地图 进阶&#xff1a; 1…

pytest入门一:用例的执行范围

从一个或多个目录开始查找&#xff0c;可以在命令行指定文件名或目录名。如果未指定&#xff0c;则使用当前目录。 测试文件以 test_ 开头或以 _test 结尾 测试类以 Test 开头 &#xff0c;并且不能带有 init 方法 测试函数以 test_ 开头 断言使用基本的 assert 即可 所有的…

steel-browser - 专为AI应用构建的开源浏览器自动化 API

Steel是一个开源浏览器 API&#xff0c;可以轻松构建与 Web 交互的 AI 应用程序和代理。您无需从头开始构建自动化基础设施&#xff0c;而是可以专注于 AI 应用程序&#xff0c;而 Steel 会处理复杂性。 2300 Stars 99 Forks 4 Issues 5 贡献者 Apache-2.0 License TypeScript …

前端H5移动端基础框架模板 :Vue3 + Vite5 + Pinia + Vant4 + Sass + 附源码

技术栈选用 Vue3 Vite5 Pinia Vant4 Sass 源码地址&#xff1a; git clone https://gitee.com/gaiya001/h5-APP.git1. 1.vite.config.js文件配置 ** import { defineConfig } from vite // 导入 Vite 的配置函数 import vue from vitejs/plugin-vue // 导入 Vue 插件 i…

继电器控制与C++编程:实现安全开关控制的技术分享

在现代生活中,继电器作为一种重要的电气控制元件,在电气设备的安全控制中起到了至关重要的作用。通过低电流控制高电流,继电器能够有效地隔离控制电路与被控设备,从而保障使用者的安全。本项目将介绍如何通过树莓派Pico与继电器模块结合,使用C++编程实现继电器的控制。 一…

Stable Diffusion Controlnet常用控制类型解析与实战课程 5

本节内容&#xff0c;是stable diffusion controlnet常用控制类型与实战的第5节课程。在前面几期课程中&#xff0c;我们已经陆续学习了controlnet的多种控制类型&#xff0c;本节课程&#xff0c;我们将继续讲解revision&#xff0c;instructp2p&#xff0c;ip-adapter&#x…

spark如何自定义函数

UDF&#xff1a;一对一的函数【User Defined Functions】 substr、split、concat、instr、length、from_unixtime UDAF&#xff1a;多对一的函数【User Defined Aggregation Functions】 聚合函数 count、sum、max、min、avg、collect_set/list UDTF&#xff1a;一对多的函…

linux网络编程 | c | select实现多路IO转接服务器

poll实现多路IO转接服务器 基于该视频完成 04-poll函数实现服务器_哔哩哔哩_bilibili 通过响应式–多路IO转接实现 要求&#xff1a;能看懂看&#xff0c;看不懂也没啥大事&#xff0c;现在基本都用epoll代替了 大家看视频思路吧&#xff0c;代码就是从讲义里面copy了一份…

数据结构(顺序表)JAVA方法的介绍

前言 在 Java 中&#xff0c;集合类&#xff08;Collections&#xff09;是构建高效程序的核心组件之一&#xff0c;而 List 接口作为集合框架中的重要一员&#xff0c;是一个有序、可重复的元素集合。与 Set 接口不同&#xff0c;List 保证了元素的顺序性&#xff0c;并允许存…

HTML+CSS+Vue3的静态网页,免费开源,可当作作业使用

拿走请吱一声&#xff0c;点个关注吧&#xff0c;代码如下&#xff0c;网页有移动端适配 HTML <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width…

内网穿透讲解

什么是内网穿透 内网穿透是一种网络技术&#xff0c;它允许外网或者其他局域网的用户来访问这个局域网的服务器资源&#xff0c;让资源的利用率更高&#xff0c;更加灵活&#xff0c;但是也要确保网络安全。 工作原理 如果你在公司&#xff0c;但是你需要使用到你家里的那台电…

Harmonyos之深浅模式适配

Harmonyos之换肤功能 概述实现原理颜色适配颜色资源配置工具类编写界面代码编写适配效果 概述 深色模式&#xff08;Dark Mode&#xff09;又称之为暗色模式&#xff0c;是与日常应用使用过程中的浅色模式&#xff08;Light Mode&#xff09;相对应的一种UI主题。 换肤功能应…

蓝桥杯刷题——day4

蓝桥杯刷题——day4 题目一题干题目解析代码 题目二题干题目解析代码 题目一 题干 小蓝和朋友们在玩一个报数游戏。由于今年是2024 年&#xff0c;他们决定要从小到大轮流报出是20或24倍数的正整数。前10个被报出的数是&#xff1a;20,24,40,48,60,72,80,96,100,120。请问第2…