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

看动画轻松理解时间复杂度(二)

 

 

上篇文章讲述了与复杂度有关的大 O 表示法和常见的时间复杂度量级,这篇文章来讲讲另外几种复杂度: 递归算法的时间复杂度(recursive algorithm time complexity),最好情况时间复杂度(best case time complexity)、最坏情况时间复杂度(worst case time complexity)、平均时间复杂度(average case time complexity)和均摊时间复杂度(amortized time complexity)。

递归算法的时间复杂度

如果递归函数中,只进行一次递归调用,递归深度为depth;

在每个递归的函数中,时间复杂度为T;

则总体的时间复杂度为O(T * depth)

在前面的学习中,归并排序 与 快速排序 都带有递归的思想,并且时间复杂度都是O(nlogn) ,但并不是有递归的函数就一定是 O(nlogn) 级别的。从以下两种情况进行分析。

① 递归中进行一次递归调用的复杂度分析

二分查找法

 1int binarySearch(int arr[], int l, int r, int target){
2    if( l > r ) return -1;
3
4    int mid = l + (r-l)/2; 
5    if( arr[mid] == target ) return mid;  
6    else if( arr[mid] > target ) 
7    return binarySearch(arr, l, mid-1, target);    // 左边 
8    else
9    return binarySearch(arr, mid+1, r, target);   // 右边

10}

比如在这段二分查找法的代码中,每次在 [ l , r ] 范围中去查找目标的位置,如果中间的元素 arr[mid] 不是 target,那么判断 arr[mid]是比 target 大 还是 小 ,进而再次调用 binarySearch这个函数。

在这个递归函数中,每一次没有找到target时,要么调用 左边 的 binarySearch函数,要么调用 右边 的 binarySearch函数。也就是说在此次递归中,最多调用了一次递归调用而已。根据数学知识,需要log2n次才能递归到底。因此,二分查找法的时间复杂度为 O(logn)。

求和

1int sum (int n) {
2  if (n == 0) return 0;
3  return n + sum( n - 1 )
4}

在这段代码中比较容易理解递归深度随输入 n 的增加而线性递增,因此时间复杂度为 O (n)。

求幂

1//递归深度:logn
2//时间复杂度:O(logn)
3double powdouble x, int n){
4  if (n == 0) return 1.0;
5
6  double t = pow(x,n/2);
7  if (n %2) return x*t*t;
8  return t * t;
9}

递归深度为 logn,因为是求需要除以 2 多少次才能到底。

② 递归中进行多次递归调用的复杂度分析

递归算法中比较难计算的是多次递归调用。

先看下面这段代码,有两次递归调用。

1// O(2^n) 指数级别的数量级,后续动态规划的优化点
2int f(int n){
if (n == 0) return 1;
return f(n-1) + f(n - 1);
5}

递归树中节点数就是代码计算的调用次数。

比如 当 n = 3 时,调用次数计算公式为

1 + 2 + 4 + 8 = 15

一般的,调用次数计算公式为

2^0 + 2^1 + 2^2 + …… + 2^n
= 2^(n+1) - 1
= O(2^n)

与之有所类似的是 归并排序 的递归树,区别点在于

  • 1. 上述例子中树的深度为 n,而 归并排序 的递归树深度为logn
  • 2. 上述例子中每次处理的数据规模是一样的,而在 归并排序 中每个节点处理的数据规模是逐渐缩小的

因此,在如 归并排序 等排序算法中,每一层处理的数据量为 O(n) 级别,同时有 logn 层,时间复杂度便是 O(nlogn)。

最好、最坏情况时间复杂度


最好、最坏情况时间复杂度指的是特殊情况下的时间复杂度。

 

动图表明的是在数组 array 中寻找变量 x 第一次出现的位置,若没有找到,则返回 -1;否则返回位置下标。

1int find(int[] array, int n, int x) {
2  for (  int i = 0 ; i < n; i++) {
3    if (array[i] == x) {
4        return i;
5        break;
6    }
7  }
8  return -1;
9}

在这里当数组中第一个元素就是要找的 x 时,时间复杂度是 O(1);而当最后一个元素才是 x 时,时间复杂度则是 O(n)。

最好情况时间复杂度就是在最理想情况下执行代码的时间复杂度,它的时间是最短的;最坏情况时间复杂度就是在最糟糕情况下执行代码的时间复杂度,它的时间是最长的。

平均情况时间复杂度

最好、最坏时间复杂度反应的是极端条件下的复杂度,发生的概率不大,不能代表平均水平。那么为了更好的表示平均情况下的算法复杂度,就需要引入平均时间复杂度。

平均情况时间复杂度可用代码在所有可能情况下执行次数的加权平均值表示。

还是以 find 函数为例,从概率的角度看, x 在数组中每一个位置的可能性是相同的,为 1 / n。那么,那么平均情况时间复杂度就可以用下面的方式计算:

((1 + 2 + … + n) / n + n) / 2 = (3n + 1) / 4

find 函数的平均时间复杂度为 O(n)。

均摊复杂度分析

我们通过一个动态数组的 push_back 操作来理解 均摊复杂度

 1template <typename T>
2class MyVector{
3private:
4    T* data;
5    int size;       // 存储数组中的元素个数
6    int capacity;   // 存储数组中可以容纳的最大的元素个数
7    // 复杂度为 O(n)
8    void resize(int newCapacity){
9        T *newData = new T[newCapacity];
10        for( int i = 0 ; i < size ; i ++ ){
11              newData[i] = data[i];
12            }
13        data = newData;
14        capacity = newCapacity;
15    }
16public:
17    MyVector(){
18        data = new T[100];
19        size = 0;
20        capacity = 100;
21    }
22    // 平均复杂度为 O(1)
23    void push_back(T e){
24        if(size == capacity)
25            resize(2 * capacity);
26        data[size++] = e;
27    }
28    // 平均复杂度为 O(1)
29    pop_back(){
30        size --;
31        return data[size];
32    }
33
34};

push_back实现的功能是往数组的末尾增加一个元素,如果数组没有满,直接往后面插入元素;如果数组满了,即 size == capacity ,则将数组扩容一倍,然后再插入元素。

例如,数组长度为 n,则前 n 次调用 push_back 复杂度都为 O(1) 级别;在第 n + 1 次则需要先进行 n 次元素转移操作,然后再进行 1 次插入操作,复杂度为 O(n)。

因此,平均来看:对于容量为 n 的动态数组,前面添加元素需要消耗了 1 * n 的时间,扩容操作消耗 n 时间 ,
总共就是 2 * n 的时间,因此均摊时间复杂度为 O(2n / n) = O(2),也就是 O(1) 级别了。

可以得出一个比较有意思的结论:一个相对比较耗时的操作,如果能保证它不会每次都被触发,那么这个相对比较耗时的操作,它所相应的时间是可以分摊到其它的操作中来的。

原文链接:看动画轻松理解时间复杂度(二)

转载于:https://www.cnblogs.com/fivestudy/p/10123842.html

相关文章:

  • finance1:专业词汇
  • 周末去面试,进去 5 分钟就出来了…
  • Automatic Login Using sshpass
  • 数据库3
  • oracle数据库日常维护手册
  • Bodymovin:Bodymovin和Lottie:把AE动画转换成HTML5/Android/iOS原生动画
  • veterbi
  • 数算运算符
  • Solaris 11 配置IP地址
  • javaweb数据库编程代码详细讲解
  • CSS 简介/特点/优势/给特定浏览器提供不同样
  • 使用Puppeteer进行数据抓取(五)——快速调试
  • 000. 规范类的设计(ing)
  • 面试题8:二叉树的下一个结点
  • pcntl_exec()
  • ECMAScript6(0):ES6简明参考手册
  • JSDuck 与 AngularJS 融合技巧
  • leetcode386. Lexicographical Numbers
  • VuePress 静态网站生成
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 记一次删除Git记录中的大文件的过程
  • 简单数学运算程序(不定期更新)
  • 项目实战-Api的解决方案
  • 一些关于Rust在2019年的思考
  • 继 XDL 之后,阿里妈妈开源大规模分布式图表征学习框架 Euler ...
  • 完善智慧办公建设,小熊U租获京东数千万元A+轮融资 ...
  • # centos7下FFmpeg环境部署记录
  • # 透过事物看本质的能力怎么培养?
  • #git 撤消对文件的更改
  • #Js篇:单线程模式同步任务异步任务任务队列事件循环setTimeout() setInterval()
  • #调用传感器数据_Flink使用函数之监控传感器温度上升提醒
  • (+3)1.3敏捷宣言与敏捷过程的特点
  • (3)(3.2) MAVLink2数据包签名(安全)
  • (第27天)Oracle 数据泵转换分区表
  • (定时器/计数器)中断系统(详解与使用)
  • (附源码)springboot工单管理系统 毕业设计 964158
  • (个人笔记质量不佳)SQL 左连接、右连接、内连接的区别
  • (十三)Flask之特殊装饰器详解
  • (学习日记)2024.01.19
  • (一)使用IDEA创建Maven项目和Maven使用入门(配图详解)
  • (转)GCC在C语言中内嵌汇编 asm __volatile__
  • (转)Oracle 9i 数据库设计指引全集(1)
  • (轉貼) 蒼井そら挑戰筋肉擂台 (Misc)
  • .locked1、locked勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .NET delegate 委托 、 Event 事件
  • .Net Redis的秒杀Dome和异步执行
  • .Net 代码性能 - (1)
  • .NET 反射的使用
  • .NET基础篇——反射的奥妙
  • .NET开源的一个小而快并且功能强大的 Windows 动态桌面软件 - DreamScene2
  • .NET框架设计—常被忽视的C#设计技巧
  • .secret勒索病毒数据恢复|金蝶、用友、管家婆、OA、速达、ERP等软件数据库恢复
  • /bin/bash^M: bad interpreter: No such file or directory
  • @Not - Empty-Null-Blank
  • @private @protected @public