数据结构-复杂度

复杂度

  • 1.数据结构
    • 1.1算法
  • 2.算法效率
    • 2.1复杂度的概念
  • 3.时间复杂度
    • 3.1大O渐进表示法
    • 3.2时间复杂度计算示例
      • 3.2.1 示例1
      • 3.2.2 示例2
      • 3.2.3 示例3
      • 3.2.4 示例4
      • 3.2.5 示例5:
      • 3.2.6 示例6
      • 3.2.7 示例7
  • 4.空间复杂度
    • 4.1.1 示例1
      • 4.1.2 示例2
  • 5.常见复杂度对比
  • 6.复杂度算法题
    • 6.1旋转数组

1.数据结构

数据结构是计算机存储、组织数据的方式,指相互之间存在一种或多种特定关系的数据元素的集合。没有一种单一的数据结构对所有用途都有用,所以我们要学各式各样的数据结构。

不仅能存储数据,还要能够管理数据。

1.1算法

算法就是定义良好的计算过程,他取一个或一组的值为输入,并产生出一个或一组值作为输出。

那么怎么样的算法算是好的呢?好算法是用什么来衡量的?

我们看看下面的代码:

#include<stdio.h>
int main()
{int t1 = clock();//表示计算代码当前所用时间for (int i = 0; i < 100000; i++){for (int j = 1; j < 10000; j++){int a = 1;}}int t2 = clock();printf("%d\n", t2 - t1);return 0;
}

上面这个代码运行结果是不同的,这和电脑的配置是相关的。所以看算法的执行时间是不行的。

2.算法效率

那么如何衡量一个算法的好坏呢?
我们来观察一个案例:
https://leetcode.cn/problems/rotate-array/description/

void rotate(int* nums, int numsSize, int k) {while(k--)//k有几次轮转几次。{int end = nums[numsSize-1];//将最后一个元素拿出来for(int i = numsSize - 1;i > 0 ;i--){nums[i] = nums[i-1];//将前一个元素放到后一个}nums[0] = end;//将拿出来的最后一个元素,放到前面。}
}

解释:
输入:num = [1,2,3,4,5,6,7],k = 3
输出:[5,6,7,1,2,3,4]
向后轮转1步:[7,1,2,3,4,5,6]
向后轮转2步:[6,7,1,2,3,4,5]
向后轮转3步:[5,6,7,1,2,3,4]

思路:循环k次将数组所有元素向后移动一位
在这里插入图片描述
点击执行可以通过,然而点击提交却无法通过,那该如何衡量其好与坏呢?

2.1复杂度的概念

算法在编写成可执行程序后,运行时需要耗费时间资源和空间(内存)资源。因此衡量一个算法的好坏,一般是从时间和空间两个维度来衡量的,即时间复杂度和空间复杂度。

时间复杂度主要衡量一个算法的运行快慢,而空间复杂度主要衡量一个算法运行所需要的额外空间。在计算机发展的早期,计算机的存储容量很小。所以对空间复杂度很是在乎。但是经过计算机行业的迅速发展,计算机的存储容量已经达到;额很高的程度。所以我们如今已经不需要再特别关注一个算法的空间复杂度。

3.时间复杂度

定义:在计算机科学中,算法的时间复杂度是一个函数式T(N),N是影响时间复杂度的输入条件,它定量描述了该算法的运行时间。时间复杂度是衡量程序的时间效率,那么为什么不去计算程序的运行时间呢?

1.因为程序运行时间和编译环境和运行机器的配置都有关系,比如同一个算法程序,用一个老编译器进行编译和新的编译器编译,在同样机器下运行时间不同。
~
2.同一个算法程序,用一个老底配置和高配置机器,运行时间也不同。
~
3并且时间只能程序写好后测试,不能写程序前通过理论思想计算评估。

这个T(N)函数式计算了程序的执行次数。那么我们通过程序代码或者理论思想计算出程序的执行次数的函数式T(N),假设每句指令执行时间基本一样(实际中有差别,但是微乎其微),那么执行次数和运行时间就是等比正相关,这样也脱离了具体的编译运行环境。执行次数就可以代表程序时间效率的优劣。比如解决一个问题的算法a程序T(N),算法b程序T(N)= N^2,那么算法a的效率一定优于算法b。

影响时间复杂度的条件有:每条语句的执行时间*每条语句的执行次数

每条语句的执行时间(无法给出准确数据;给出结论:每条语句的执行时间即使有差别但是微乎其微,可以忽略不计,认为每条语句的执行时间是相同的。)

案例:
请计算⼀下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; } 
}

根据代码中的执行的基本操作次数:

T (N) = N + 2*N + 10

可以看出对结果影响最大的是N^2。

时间复杂度只能用来表示输入条件对事件的影响趋势

实际中我们计算时间复杂度时,计算的也不是程序的精确的执行次数,精确执行次数计算起来还是很麻烦的(不同的一句程序代码,编译出的指令条数是不一样的),计算出精确的执行次数意义也不大,因为我们计算时间复杂度只是想比较算法程序的增长量级,也就是当N不断变大时T(N)的差别,上面我们已经看到了当N不断变大时常数和低阶项对结果的影响很小,所以我们只需要计算程序能代表增长量级的大概执行次数,复杂度的表示通常使用大O的渐进表示法

3.1大O渐进表示法

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

推导大O阶规则:
1.时间复杂度函数式T(N)中,只保留最高阶项,去掉那些低阶项,因为当N不断变大时,低阶项对结果影响越来越小,当N无穷大时,就可以忽略不计了
~
2.如果最高阶项存在且不是1,则去除这个项目的常熟系数,因为当N不断变大,这个系数对结果影响越来越小,当N无穷大时,就可以忽略不计了。
~
3.T(N)中如果没有N相关的项目,只有常数项,用常数1取代所有加法常数。

3.2时间复杂度计算示例

3.2.1 示例1

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

Func2执行的基本操作次数:
T(N)= 2*N + 10
根据推导规则第3条得出:
复杂度为:O(N)

数据结构中只考虑变化对时间复杂度的影响。

3.2.2 示例2

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

Func3执行的基本操作次数:
T(N) = M + N
有两个可变条件。需要进一步讨论:
M >> N : O(M)
N >> M : O(N)
N == M : O(M)或者O(N)

3.2.3 示例3

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

T(N)= 100
根据推导规则第1条得出
时间复杂度为:O(1)
这里的1不是执行一次,而是表示常数

无论常数是多少,常数对时间的增长趋势没有任何影响

3.2.4 示例4

const char * strchr ( const char 
* str, int character)
{const char* p_begin = s;while (*p_begin != character){if (*p_begin == '\0')return NULL;p_begin++;
}return p_begin;}

假设字符串长度为n:
查找的是前面的字符:查找常数次
查找的是后面的字符:查找n次
查找的是中间的字符:查找n/2,就是n次。

对于当前的时间复杂度来说,我们要划分为不同的场景:

查找的是前面的字符,时间更少,那么我们的时间复杂度就更优一些,就称之为最好的情况。所以上面的三种情况可以被分为:最好情况、最坏情况和平均情况。

因此时间复杂度可以写为:
最好情况:O(1)
最坏情况:O(N)
平均情况:O(N)

总结
通过上面我们会发现,有些算法的时间复杂度存在最好、平均和最坏情况
最坏情况:任意输入规模的最大运行次数(上界)
平均情况:任意输入规模的期望运行次数
最好情况:任意输入规模的最小运行次数(下界)

大O的渐进表示法在实际中一般情况关注的是算法的上界,也就是最坏运行情况。

3.2.5 示例5:

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)若数组有序,则:
T(N)= N
(2)若数组有序且为降序,则:
T(N)=N*(N+1)/2

当数组有序的时候 只需要比较n-1
即冒泡排序时间复杂度最好的情况为O(n)
最差的情况,时间复杂度为:O(n^2)
只要有一个是乱序,就是最差的情况

3.2.6 示例6

void func5(int n)
{int cnt = 1;while (cnt < n){cnt *= 2;}
}

当n=2时,执行次数为1
当n=4时,执行次数为2
当n=16时,执行次数为4
假设执⾏次数为x ,则2^x = n
因此执⾏次数:x = log n

因此:时间复杂度最差情况为:O(log2n)

注意:

注意log2n、logn、lgn的表示
当n接近无穷大时,底数的大小对结果影响不大。因此,一般情况下不管底数是多少都可以省略不写,即可以表示为logn

3.2.7 示例7

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

调用一次Fac函数的时间复杂度为O(1)
而在Fac函数中,存在n次递归函数调用Fac函数
因此:
时间复杂度为:O(n)

4.空间复杂度

空间复杂度也是一个数学表达式,是对一个算法在运行过程中因为算法的需要额外临时开辟的空间

空间复杂度不是程序占用了多少bytes的空间,因为常规情况每个对象大小差异不会很大,所以空间复杂度算的是变量的个数。

空间复杂度计算规则基本跟时间复杂度类似,也使用大O渐进表示法

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

4.1.1 示例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; } 
}

函数栈帧在编译期间已经确定好了,只需要关注函数在运行时额外申请的空间。

BubbleSort额外申请的空间有exchange等有限个局部变量,使用了常数个额外空间

在函数体里面是运行时确定申请空间。

因此空间复杂度为O(1)

4.1.2 示例2

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

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

因此空间复杂度为:O(N)

时间复杂度和空间复杂度存在部分相同,因为,时间复杂度求解的是执行次数,空间复杂度申请的是空间。

5.常见复杂度对比

从左到右复杂度越优。
在这里插入图片描述
在这里插入图片描述

6.复杂度算法题

6.1旋转数组

https://leetcode.cn/problems/rotate-array/description/

思路一:

时间复杂度O(n^2)
循环k次将数组所有元素向后移动一位(不通过)

void rotate(int* nums, int numsSize, int k) {while(k--){int end = nums[numsSize-1];for(int i = numsSize - 1;i > 0 ;i--){nums[i] = nums[i-1];}nums[0] = end;}
}

时间复杂度为O(n^2)
空间复杂度为O(1)

该代码超出时间复杂度

思路二:

空间复杂度O(n)
申请新数组空间,先将后k个数据放到新数组中,在将剩下的数据挪到新数组中

void rotate(int* nums, int numsSize, int k) 
{int newArr[numsSize];for (int i = 0; i < numsSize; ++i) {newArr[(i + k) % numsSize] = nums[i];}for (int i = 0; i < numsSize; ++i) {nums[i] = newArr[i];}
}

在这里插入图片描述
时间复杂度 O(N)
空间复杂度 O(N)

以空间换时间的方式来提高算法性能

思路三:

空间复杂度O(1)

• 前n-k个逆置:4 3 2 1 5 6 7

• 后k个逆置:4 3 2 1 7 6 5

• 整体逆置:5 6 7 1 2 3 4

void reverse(int* nums,int begin,int end)
{while(begin<end){int tmp = nums[begin];nums[begin] = nums[end];nums[end] = tmp;begin++;end--;}
}
void rotate(int* nums, int numsSize, int k)
{k = k%numsSize;reverse(nums,0,numsSize-k-1);reverse(nums,numsSize-k,numsSize-1);reverse(nums,0,numsSize-1);
}

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

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

相关文章

【重学 MySQL】六十七、解锁检查约束,守护数据完整性

【重学 MySQL】六十七、解锁检查约束&#xff0c;守护数据完整性 检查约束的基本概念检查约束的语法检查约束的使用场景注意事项示例 在MySQL中&#xff0c;检查约束&#xff08;CHECK&#xff09;是一种用于确保表中数据满足特定条件的约束。 检查约束的基本概念 检查约束用…

考研前所学c语言02(2024/10/16)

1.一个十进制的数转化为二进制的就是不断除二取余&#xff0c;得到的余数从下到上取 比如123&#xff1a; 结果为&#xff1a; 同理其他的十进制转八进制&#xff0c;十六进制就除八&#xff0c;除十六即可 再比如123转十六进制&#xff1a; 因为余数是11&#xff0c;十六进…

【JavaEE初阶】深入理解网络编程—使用UDP协议API实现回显服务器

前言 &#x1f31f;&#x1f31f;本期讲解关于TCP/UDP协议的原理理解~~~ &#x1f308;感兴趣的小伙伴看一看小编主页&#xff1a;GGBondlctrl-CSDN博客 &#x1f525; 你的点赞就是小编不断更新的最大动力 &#x1f386;那么废话不…

从0开始深度学习(12)——多层感知机的逐步实现

依然以Fashion-MNIST图像分类数据集为例&#xff0c;手动实现多层感知机和激活函数的编写&#xff0c;大部分代码均在从0开始深度学习&#xff08;9&#xff09;——softmax回归的逐步实现中实现过 1 读取数据 import torch from torchvision import transforms import torchv…

查找与排序-交换排序

交换排序是基于“比较”和“交换”两种操作来实现的排序方法 。 由于选择“比较”的基准元素不同&#xff0c;可将交换排序分为以下两种&#xff1a; 冒泡排序快速排序 一、冒泡排序 1.冒泡排序基本思想 因为其实现与气泡从水中往上冒的过程类似而得名。 每一趟的…

基于SpringBoot+Vue+uniapp微信小程序的垃圾分类系统的详细设计和实现(源码+lw+部署文档+讲解等)

项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不是配置文件。Spring Boot 通过自动化配置和约…

Redux (八) 路由React-router、嵌套路由、路由传参、路由懒加载

文章目录 一、React-Router的基本使用1. 安装及基本使用(路由映射配置)2. 路由跳转Link与NavLink3. Navigate导航4. 处理路径不存在的情况 二、嵌套路由三、手动跳转 (类似编程式路由导航)1. 函数式组件2. 类组件实现手动跳转 四、路由传参1. 路径设置占位符(params)2. search传…

Java面试指南:Java基础介绍

这是《Java面试指南》系列的第1篇&#xff0c;本篇主要是介绍Java的一些基础内容&#xff1a; 1、Java语言的起源 2、Java EE、Java SE、Java ME介绍 3、Java语言的特点 4、Java和C的区别和联系&#xff1f; 5、面向对象和面向过程的比较 6、Java面向对象的三大特性&#xff1a…

leetcode30:串联所有单词的字串

给定一个字符串 s 和一个字符串数组 words。 words 中所有字符串 长度相同。 s 中的 串联子串 是指一个包含 words 中所有字符串以任意顺序排列连接起来的子串。 例如&#xff0c;如果 words ["ab","cd","ef"]&#xff0c; 那么 "abcdef…

1. 解读DLT698.45-2017通信规约--预连接响应

国家电网有限公司企业标准&#xff0c;面向对象的用电信息数据交换协议DLT698.45-2017 为提高用电信息采集系统的业务适应性、采集效率、安全性和数据溯源性&#xff0c;规范用电信息数据交换协议的通信架构、数据链路层、应用层、接口类与对象标识&#xff0c;制定本标准。 …

Linux系统:(Linux系统概述与安装)

硬件计算机硬件是指计算机系统中所有物理部件的总称。包括计算机主机、显示器、键盘、鼠标、内存、硬盘、处理器、主板等等。这些硬件部件是计算机系统运行的基础 不管是电脑系统(个人电脑、服务器等)、还是移动端操作系统(手机、平板等)。它的功能就是做为用户和硬件之间的桥梁…

前端求职简历-待补充

当然可以&#xff0c;针对大厂的前端岗位&#xff0c;一个吸引人的简历应该突出你的技术能力、项目经验、教育背景以及任何能体现你学习能力和团队协作能力的证明。以下是一个简历大纲示例&#xff0c;你可以根据自己的实际情况进行调整&#xff1a; 个人信息 姓名联系方式&a…

图文深入介绍oracle资源管理(续)

1. 引言&#xff1a; 本文将承接上篇继续深入介绍oracle资源管理。本文重点介绍如何使用oracle资源管理器管理好DB。 2. 资源管理器&#xff1a; 可以使用图形界面 OEM$或命令行调用 DBMS RESOURCE MANAGER 程序包的过程进行数据库资源管理。 调用资源管理器的先决条件&…

瑞数后缀加密怎么处理

前言&#xff1a; 瑞数我们经常补环境通过&#xff0c;但是遇到瑞数后缀不知道怎么处理 就拿瑞数4来讲 解决方法&#xff1a; &#xff08;1&#xff09;传明文加密参数 一般情况&#xff0c;我们传明文加密参数也能访问 &#xff08;2&#xff09;再补环境基础调用open …

基于stm32的4G模块点灯实验

led模块功能封装 #include "led.h" #include "sys.h"//初始化GPIO函数 void led_init(void) {GPIO_InitTypeDef gpio_initstruct;//打开时钟__HAL_RCC_GPIOB_CLK_ENABLE();//调用GPIO初始化函数gpio_initstruct.Pin GPIO_PIN_8 | GPIO_PIN_9;gpio_inits…

排序算法 —— 直接插入排序

目录 1.直接插入排序的思想 2.直接插入排序的实现 实现分析 实现代码 3.直接插入排序的分析 时间复杂度分析 空间复杂度分析 稳定性 1.直接插入排序的思想 直接插入排序的思想就是把待排序的元素按其关键码值的大小依次插入到一个已经排好序的有序序列中&#xff0c…

pycharm调试带参数命令行的程序

点击 run configuration 点击加号&#xff0c;选择python name填写程序名字&#xff0c;script填写程序路径&#xff0c;下一行填写传入的参数 点击apply&#xff0c;再点ok&#xff0c;即可debug 参考&#xff1a; pycharm 调试模式下命令行参数的传递_pycharm debug传参 ya…

项目实战:构建 effet.js 人脸识别交互系统的实战之路

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀构建 effet.js &#x1f4d2;1. 什么是effet.js&#x1f4dc;2. 为什么需要使用effet.js&#x1f4dd;3. effet.js的功能&#x1f4da;4. 使用…

【项目案例】-音乐播放器-Android前端实现-Java后端实现

精品专题&#xff1a; 01.C语言从不挂科到高绩点 https://blog.csdn.net/yueyehuguang/category_12753294.html?spm1001.2014.3001.5482https://blog.csdn.net/yueyehuguang/category_12753294.html?spm1001.2014.3001.5482 02. SpringBoot详细教程 https://blog.csdn.ne…

HTML之表单设计

1、HTML表单 HTML表单是用于收集用户输入的信息&#xff0c;并将用户输入的内容信息传到后台服务器中。 表单是通过form标签实现。 特别注意&#xff1a;如果一些内容提交后&#xff0c;没有将内容提交给后台服务器&#xff0c;那么需要添加一个name属性&#xff0c;语法&am…