当前位置: 首页 > news >正文

算法复杂度的简单介绍

算法复杂度是衡量算法执行效率和资源消耗的指标,通常分为时间复杂度空间复杂度。时间复杂度评估算法执行所需时间随输入规模的变化,空间复杂度评估算法占用内存的增长情况。复杂度通常用大O符号来表示,它描述了最坏情况下的增长速率。

1. 时间复杂度

时间复杂度表示算法执行所需时间随输入规模 nnn 的变化关系。常见的时间复杂度如下(从快到慢):

a. 常数时间:O(1)

  • 不管输入大小如何,算法总是执行固定的操作。
  • 示例:数组中访问某个元素。
    int element = array[5]; // O(1) 

b. 对数时间:O(log n)

  • 每次迭代减少问题规模的某个倍数,通常是二分法等算法。
  • 示例:二分查找。
    int binarySearch(int arr[], int size, int target) {int left = 0, right = size - 1;while (left <= right) {int mid = left + (right - left) / 2;if (arr[mid] == target)return mid;else if (arr[mid] < target)left = mid + 1;elseright = mid - 1;}return -1;
    }
     

c. 线性时间:O(n)

  • 算法的时间复杂度随着输入大小线性增长。
  • 示例:遍历一个数组。
    for (int i = 0; i < n; i++) { // O(n) } 

d. 线性对数时间:O(n log n)

  • 比线性复杂度稍慢,常见于高效的排序算法,如归并排序、快速排序。
  • 示例:归并排序。
    void mergeSort(int arr[], int n) { if (n > 1) { // Divide the array into two halves mergeSort(arr, n / 2); mergeSort(arr + n / 2, n - n / 2); // Merge the two halves } } 

e. 平方时间:O(n²)

  • 算法的时间复杂度随着输入规模的平方增长,常见于嵌套的循环。
  • 示例:冒泡排序。
  • void bubbleSort(int arr[], int n) {for (int i = 0; i < n - 1; i++) {for (int j = 0; j < n - i - 1; j++) {if (arr[j] > arr[j + 1]) {// swap}}}
    }

f. 立方时间:O(n³)

  • 时间复杂度与输入规模的立方成比例,通常出现在三重嵌套循环中。
  • 示例:矩阵乘法的朴素算法。

g. 指数时间:O(2^n)

  • 输入规模每增加1,执行时间翻倍,通常是解决组合问题的递归算法。
  • 示例:递归解决斐波那契数列。
    int fibonacci(int n) {if (n <= 1) return n;return fibonacci(n - 1) + fibonacci(n - 2);  // O(2^n)
    }
     

h. 阶乘时间:O(n!)

  • 输入规模每增加1,执行时间呈阶乘增长,常见于排列组合问题。
  • 示例:解决N皇后问题。

2. 空间复杂度

空间复杂度评估算法所需的内存随输入规模的变化情况。它表示程序在运行过程中需要的额外存储空间。

  • O(1):常数空间,不管输入规模如何,所需额外空间是固定的。例如,交换两个变量的算法。
  • O(n):线性空间,所需的额外空间随着输入规模线性增加。例如,使用一个数组存储输入数据的副本。
  • O(n²):所需的存储空间与输入的平方成正比,通常出现在矩阵相关的问题中。

3. 常见算法的时间复杂度表

算法类型最佳情况平均情况最坏情况
冒泡排序O(n)O(n²)O(n²)
选择排序O(n²)O(n²)O(n²)
插入排序O(n)O(n²)O(n²)
归并排序O(n log n)O(n log n)O(n log n)
快速排序O(n log n)O(n log n)O(n²)
二分查找O(1)O(log n)O(log n)
线性查找O(1)O(n)O(n)

4. 大O符号的简化原则

在表示时间复杂度时,我们通常关注输入规模 nnn 增长时的趋势,因此忽略常数项和低次项。例如:

  • 如果一个算法的复杂度是 5n+35n + 35n+3,我们会简化为 O(n)O(n)O(n)。
  • 如果复杂度是 n2+nn^2 + nn2+n,我们会简化为 O(n2)O(n^2)O(n2)。

5. 实际应用中的复杂度权衡

虽然理论上高效的算法(如 O(n log n) 的排序算法)往往表现优异,但在某些场景下,低复杂度的算法未必比高复杂度的算法快。例如,当输入规模较小时,冒泡排序(O(n²))可能比快速排序(O(n log n))更快。因此,选择算法时需要考虑实际数据规模硬件环境

6. 总结

  • 时间复杂度衡量算法执行时间随输入规模的增长而变化。
  • 空间复杂度评估算法所需的存储空间随输入规模的变化。
  • 常见的时间复杂度包括:O(1), O(log n), O(n), O(n log n), O(n²) 等,越高阶的复杂度,随着输入规模增长,算法性能下降越快。
  • 编写高效算法时应考虑优化时间和空间复杂度,在不同的场景中合理选择算法。

 

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 程序的格式框架与缩进
  • Unity之获取Avpro视频画面并在本地创建缩略图
  • 手机扬声器音量总是不够大?试试“扬声器助推器”吧
  • 停车场小程序如何实现分账功能?
  • WIN11 ESP32 IDF + VSCODE 环境搭建[教程向]
  • 心法利器[118] | 向量检索组件(含代码)
  • [论文笔记] t-SNE数据可视化
  • 数字逻辑设计基础
  • 数据结构——单链表相关操作
  • 和服务端系统的通信
  • 正则表达式之grep
  • [C#学习笔记]注释
  • 信息学奥赛初赛天天练-86-NOIP2014普及组-基础题5-球盒问题、枚举算法、单源最短路、Dijkstra算法、Bellman-Ford算法
  • 营养方案调整执行流程 第十篇
  • Spring Batch
  • cookie和session
  • fetch 从初识到应用
  • js学习笔记
  • php面试题 汇集2
  • Python3爬取英雄联盟英雄皮肤大图
  • Theano - 导数
  • vue从入门到进阶:计算属性computed与侦听器watch(三)
  • Vue小说阅读器(仿追书神器)
  • 阿里云ubuntu14.04 Nginx反向代理Nodejs
  • 初识MongoDB分片
  • 代理模式
  • 好的网址,关于.net 4.0 ,vs 2010
  • 软件开发学习的5大技巧,你知道吗?
  • 使用Swoole加速Laravel(正式环境中)
  • 好程序员web前端教程分享CSS不同元素margin的计算 ...
  • 交换综合实验一
  • 直播平台建设千万不要忘记流媒体服务器的存在 ...
  • ​Java基础复习笔记 第16章:网络编程
  • #Js篇:单线程模式同步任务异步任务任务队列事件循环setTimeout() setInterval()
  • #数据结构 笔记三
  • #我与Java虚拟机的故事#连载14:挑战高薪面试必看
  • (2022 CVPR) Unbiased Teacher v2
  • (a /b)*c的值
  • (android 地图实战开发)3 在地图上显示当前位置和自定义银行位置
  • (C#)一个最简单的链表类
  • (动手学习深度学习)第13章 计算机视觉---图像增广与微调
  • (附源码)spring boot建达集团公司平台 毕业设计 141538
  • (附源码)小程序 交通违法举报系统 毕业设计 242045
  • (含笔试题)深度解析数据在内存中的存储
  • (微服务实战)预付卡平台支付交易系统卡充值业务流程设计
  • (一)项目实践-利用Appdesigner制作目标跟踪仿真软件
  • (转)IIS6 ASP 0251超过响应缓冲区限制错误的解决方法
  • (转)关于pipe()的详细解析
  • ***通过什么方式***网吧
  • .NET Core 通过 Ef Core 操作 Mysql
  • .net core 源码_ASP.NET Core之Identity源码学习
  • .net 获取url的方法
  • .NET 中什么样的类是可使用 await 异步等待的?
  • .NET/C# 使窗口永不获得焦点
  • .NET导入Excel数据