浅识数据结构之时间复杂度

embedded/2024/10/18 8:20:20/
P. S.:以下代码均在VS2019环境下测试,不代表所有编译器均可通过。
P. S.:测试代码均未展示头文件stdio.h的声明,使用时请自行添加。

  

文章目录

  • 前言
  • 一. 时间复杂度
    • 1.1 时间复杂度的概念
    • 1.2 时间复杂度如何计算
    • 1.3 时间复杂度如何表示
  • 二. 常见复杂度举例
  • 三. 结语

前言


  在了解时间复杂度之前,先向大家科普一个大环境知识。
  当今时代的科学技术,让计算机在面对执行指令是可以以一个夸张的速度进行执行,也就是我们口中对于计算机设备的频率,下面引入一段代码:
void test01()
{int begin = clock();int x = 0;for (int i = 0; i < 100000000; i++){++x;}int end = clock();int time = end - begin;printf("x = %d\n", x);printf("time = %d ms\n",time);
}

在这里插入图片描述


  在如上代码中,我们看到即使我们需要将这个循环执行一亿次,但在计算机设备的加持下,不到一秒就出了结果,可见其执行效率之高。


一. 时间复杂度

1.1 时间复杂度的概念


  时间复杂度的定义:
  在计算机科学中,算法的时间复杂度是一个函数,它定量描述了该算法的运行时间。一个算法执行所耗费的时间,从理论上说,是不能算出来的,只有你把你的程序放在机器上跑起来,才能知道。但是我们需要每个算法都上机测试吗?是可以都上机测试,但是这很麻烦,所以才有了时间复杂度这个分析方式。一个算法所花费的时间与其中语句的执行次数成正比例,算法中的基本操作的执行次数,为算法的时间复杂度。
  更加通俗的来讲就是:找到某条基本语句与问题规模N之间的数学表达式,就是算出了该算法的时间复杂度。

1.2 时间复杂度如何计算


  我们照样以实例作为参考:
// 请计算一下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);
}

  我们将它转化为一个数学表达式

                    F(N) = N² + 2 × N + 10

  在计算时间复杂度时我们将数学表达式中的影响最大的部分摘下来,并将复杂度写作O( N² )


  这时就会有同学问了,上面不是讲到复杂度时基本语句的执行次数的问题吗?为什么上面代码中 count 和 M的定义语句不添加到其中呢?
  在前言中我们讲到,如今的计算机设备执行速率已经十分快速,对于其中的一些细枝末节的语句执行次数我们可以忽略不计,除非他的数量大到可以占到总体的一半以上,否则不添加到计算当中,此时我们就可以引入数学中的极限思想,当上述表达式中N趋近于无穷时,看其最高阶项,其他的在最高阶面前都可忽略不计,因为它太太太小了。

1.3 时间复杂度如何表示


  大O符号(Big O notation):是用于描述函数渐进行为的数学符号。
  推导大O阶方法:

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

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

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


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

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

  平均情况:任意输入规模的期望运行次数

  最好情况:任意输入规模的最小运行次数(下界)

  例如:在一个长度为N数组中搜索一个数据x

  最好情况:1次找到

  最坏情况:N次找到

  平均情况:N/2次找到

  在实际中一般情况关注的是算法的最坏运行情况,所以数组中搜索数据时间复杂度为O(N)。




二. 常见复杂度举例

示例复杂度名称
5201314O( 1 )常数阶
3n + 4O( n )线性阶
3n^2 + 4n + 5O( n² )平方阶
3 log( 2 ) n + 4O( log n )对数阶
3n*log( 2 ) n + 4O( n*log n )nlogn阶
n^3 + n^2 + 4O( n^3 )立方阶
2^nO( 2^n )指数阶

  有意思的是,在数据结构学习中,复杂度log N一般代表的都是以 2 为底的对数,只有当底数为 2 时才可以省略,反之不可以省略,但不是二的情况很少出现,因为其实用价值并不高。


三. 结语


  十分感谢您观看我的原创文章。
  本文主要用于个人学习和知识分享,学习路漫漫,如有错误,感谢指正。
  如需引用,注明地址。


http://www.ppmy.cn/embedded/21252.html

相关文章

IntelliJ IDEA 如何启用 JDK 预览特性

IntelliJ IDEA 也可以启用 JDK 的预览特性。 针对项目&#xff0c;选择项目结构。 配置是在语言结构上。 单击语言结构上的 SDK 默认&#xff0c;往下拉&#xff0c;就可以看到针对新版本的选项。 同时还可以看到那些版本是支持新特性预览的&#xff0c;那些版本是不支持新特…

HotSpot JVM 中的应用程序/动态类数据共享

0.前言 本文的目的是详细讨论 HotSpot JVM 自 JDK 1.5 以来提供的一项功能&#xff0c;该功能可以减少启动时间&#xff0c;但如果在多个 JVM 之间共享相同的类数据共享 (CDS) 存档&#xff0c;则还可以减少内存占用。 1.类数据共享 (CDS) CDS 的想法是使用特定格式将预处理…

闲话 ASP.NET Core 数据校验(一):内置数据校验

前言 所谓输入的是垃圾&#xff0c;输出也必然是垃圾&#xff0c;有多少安全问题隐藏在请求的数据中&#xff0c;所以永远不能相信来自用户端的输入。 对请求数据的合法性进行校验&#xff0c;不仅有助于提升用户界面的友好性&#xff0c;而且有助于提高后台程序的安全性和稳…

【Hadoop】- MapReduce YARN 初体验[9]

目录 提交MapReduce程序至YARN运行 1、提交wordcount示例程序 1.1、先准备words.txt文件上传到hdfs&#xff0c;文件内容如下&#xff1a; 1.2、在hdfs中创建两个文件夹&#xff0c;分别为/input、/output 1.3、将创建好的words.txt文件上传到hdfs中/input 1.4、提交MapR…

VMware-Linux切换桥接模式上网教程(超详细)

这里写目录标题 1. 虚拟机关机2. VMware 虚拟网络配置2.1 检查是否存在 VMnet02.2 修改桥接模式2.3 修改Linux虚拟机网络适配器 3. Linux 系统配置3.1 修改系统网卡配置3.1.1 配置项含义解释3.1.2 查看物理机网络信息3.3.3 修改配置 3.2 重启服务 4. 测试网络连接情况5. 注意事…

mamba 和conda 安装R包

**1. 下载miniconda3 ** wget https://mirrors.tuna.tsinghua.edu.cn/anaconda/miniconda/Miniconda3-latest-Linux-x86_64.sh这个命令是在linux终端中输入的,miniconda3管理起来更方便。 2. 安装miniconda3 sh Miniconda3-latest-Linux-x86_64.sh接下来会有一些回车(ente…

什么是vue,vue怎样使用?

Vue (读音 /vjuː/&#xff0c;类似于 view) 是一套用于构建用户界面的渐进式框架。与其它大型框架不同的是&#xff0c;Vue 被设计为可以自底向上逐层应用。Vue 的核心库只关注视图层&#xff0c;不仅易于上手&#xff0c;还便于与第三方库或既有项目整合。另一方面&#xff0…

1、Flink DataStreamAPI 概述(上)

一、DataStream API 1、概述 1&#xff09;Flink程序剖析 1.Flink程序组成 a&#xff09;Flink程序基本组成 获取一个执行环境&#xff08;execution environment&#xff09;&#xff1b;加载/创建初始数据&#xff1b;指定数据相关的转换&#xff1b;指定计算结果的存储…