当前位置:   article > 正文

关于时间复杂度分析

关于时间复杂度的分析

我们假设计算机运行一行基础代码需要执行一次运算。

  1. int aFunc(void) {
  2. printf("Hello, World!\n"); // 需要执行 1
  3. return 0; // 需要执行 1
  4. }

那么上面这个方法需要执行 2 次运算

  1. int aFunc(int n) {
  2. for(int i = 0; i<n; i++) { // 需要执行 (n + 1) 次
  3. printf("Hello, World!\n"); // 需要执行 n 次
  4. }
  5. return 0; // 需要执行 1
  6. }

这个方法需要 (n + 1 + n + 1) = 2n + 2 次运算。

我们把 算法需要执行的运算次数 用 输入大小n 的函数 表示,即 T(n) 。
此时为了 估算算法需要的运行时间 和 简化算法分析,我们引入时间复杂度的概念。

定义:存在常数 c 和函数 f(N),使得当 N >= c 时 T(N) <= f(N),表示为 T(n) = O(f(n)) 。
如图:

91c7bef9bcaeec9043fc387117923414046.jpg

 

    当 N >= 2 的时候,f(n) = n^2 总是大于 T(n) = n + 2 的,于是我们说 f(n) 的增长速度是大于或者等于 T(n) 的,也说 f(n) 是 T(n) 的上界,可以表示为 T(n) = O(f(n))。因为f(n) 的增长速度是大于或者等于 T(n) 的,即T(n) = O(f(n)),所以我们可以用 f(n) 的增长速度来度量 T(n) 的增长速度,所以我们说这个算法的时间复杂度是 O(f(n))。

    算法的时间复杂度,是用来度量算法的运行时间,记作: T(n) = O(f(n))。它表示随着 输入大小n 的增大,算法执行需要的时间的增长速度可以用 f(n) 来描述。

    显然如果 T(n) = n^2,那么 T(n) = O(n^2),T(n) = O(n^3),T(n) = O(n^4) 都是成立的,但是因为第一个 f(n) 的增长速度与 T(n) 是最接近的,所以第一个是最好的选择,所以我们说这个算法的复杂度是 O(n^2) 。

       那么当我们拿到算法的执行次数函数 T(n) 之后怎么得到算法的时间复杂度呢?

  1. 我们知道常数项对函数的增长速度影响并不大,所以当 T(n) = c,c 为一个常数的时候,我们说这个算法的时间复杂度为 O(1);如果 T(n) 不等于一个常数项时,直接将常数项省略。
  1. 比如
  2. 第一个 Hello, World 的例子中 T(n) = 2,所以我们说那个函数(算法)的时间复杂度为 O(1)。
  3. T(n) = n + 29,此时时间复杂度为 O(n)。

 

    2.我们知道高次项对于函数的增长速度的影响是最大的。n^3 的增长速度是远超 n^2 的,同时 n^2 的增长速度是远超 n 的。 同时因为要求的精度不高,所以我们直接忽略低此项。

  1. 比如
  2. T(n) = n^3 + n^2 + 29,此时时间复杂度为 O(n^3)。

 

    3.因为函数的阶数对函数的增长速度的影响是最显著的,所以我们忽略与最高阶相乘的常数。

 

  1. 比如
  2. T(n) = 3n^3,此时时间复杂度为 O(n^3)。

    综合起来:如果一个算法的执行次数是 T(n),那么只保留最高次项,同时忽略最高项的系数后得到函数 f(n),此时算法的时间复杂度就是 O(f(n))。为了方便描述,下文称此为 大O推导法。

    由此可见,由执行次数 T(n) 得到时间复杂度并不困难,很多时候困难的是从算法通过分析和数学运算得到 T(n)。对此,提供下列四个便利的法则,这些法则都是可以简单推导出来的,总结出来以便提高效率。

  1. 对于一个循环,假设循环体的时间复杂度为 O(n),循环次数为 m,则这个
    循环的时间复杂度为 O(n×m)。
  1. void aFunc(int n) {
  2. for(int i = 0; i < n; i++) { // 循环次数为 n
  3. printf("Hello, World!\n"); // 循环体时间复杂度为 O(1)
  4. }
  5. }

此时时间复杂度为 O(n × 1),即 O(n)。

 

    2.对于多个循环,假设循环体的时间复杂度为 O(n),各个循环的循环次数分别是a, b, c...,则这个循环的时间复杂度为 O(n×a×b×c...)。分析的时候应该由里向外分析这些循环。

  1. void aFunc(int n) {
  2. for(int i = 0; i < n; i++) { // 循环次数为 n
  3. for(int j = 0; j < n; j++) { // 循环次数为 n
  4. printf("Hello, World!\n"); // 循环体时间复杂度为 O(1)
  5. }
  6. }
  7. }

此时时间复杂度为 O(n × n × 1),即 O(n^2)。

 

    3.对于顺序执行的语句或者算法,总的时间复杂度等于其中最大的时间复杂度。

 

  1. void aFunc(int n) {
  2. // 第一部分时间复杂度为 O(n^2)
  3. for(int i = 0; i < n; i++) {
  4. for(int j = 0; j < n; j++) {
  5. printf("Hello, World!\n");
  6. }
  7. }
  8. // 第二部分时间复杂度为 O(n)
  9. for(int j = 0; j < n; j++) {
  10. printf("Hello, World!\n");
  11. }
  12. }

此时时间复杂度为 max(O(n^2), O(n)),即 O(n^2)。

 

        4.对于条件判断语句,总的时间复杂度等于其中 时间复杂度最大的路径 的时间复杂度

  1. void aFunc(int n) {
  2. if (n >= 0) {
  3. // 第一条路径时间复杂度为 O(n^2)
  4. for(int i = 0; i < n; i++) {
  5. for(int j = 0; j < n; j++) {
  6. printf("输入数据大于等于零\n");
  7. }
  8. }
  9. } else {
  10. // 第二条路径时间复杂度为 O(n)
  11. for(int j = 0; j < n; j++) {
  12. printf("输入数据小于零\n");
  13. }
  14. }
  15. }

    此时时间复杂度为 max(O(n^2), O(n)),即 O(n^2)。


小结:

        时间复杂度分析的基本策略是:从内向外分析,从最深层开始分析。如果遇到函数调用,要深入函数进行分析。


 

 

转载于:https://my.oschina.net/u/4167465/blog/3081906

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/842015
推荐阅读
相关标签
  

闽ICP备14008679号