如何评估代码的复杂度

代码具有两种复杂度衡量方向,一个是时间复杂度,一个是空间复杂度

一,时间复杂度
定义:如果一个问题的规模是n,解决这一问题的某一算法所需要的时间为T(n),它是n的某一函数T(n)称为这一算法的“时间复杂性”。

性质:
1,渐近时间复杂性:当输入量n逐渐加大时,时间复杂性的极限情形。

T(n)=O(f(n))

T(n)表示为时间复杂度

大O记法表示该函数具有上限

f(n)表示问题本身的规模n造成的时间消耗上的函数关系

2,推导大O阶方法

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

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

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

3,具体情况:

常数阶:顺序执行没有循环

线性阶:具有循环,时间复杂度和循环的次数有关

int i; 
for(i = 0; i < n; i++){
/*时间复杂度为O(1)的程序步骤序列*/
}
//循环n次,总的时间复杂度是O(n)

对数阶:由于每次count乘以2之后,就距离n更近了一分。 也就是说,有多少个2相乘后大于n,则会退出循环。 由2^x=n 得到x=logn。 所以这个循环的时间复杂度为O(logn)。

int count = 1; 
while (count < n){
count = count * 2;
  /*时间复杂度为O(1)的程序步骤序列*/
}

平方阶:
而对于外层的循环,不过是内部这个时间复杂度为O(n)的语句,再循环n次。所以这段代码的时间复杂度为O(n^2)。如果外循环的循环次数改为了m,时间复杂度就变为O(mXn)所以我们可以总结得出,循环的时间复杂度等于循环体的复杂度乘以该循环运行的次数。

例1:
int i, j; for(i = 0; i < n; i++){
  for(j = 0; j < n; j++){
/*时间复杂度为O(1)的程序步骤序列*/
  }
}//内外循环无关,时间复杂度内外相乘
例2:
int i, j; 

for(i = 0; i < n; i++){

  for(j = i; j < n; j++){ /*注意j = i而不是0*/

  /*时间复杂度为O(1)的程序步骤序列*/

  }



}//内外循环有关系,需要逐一相加,得出关系式,并且使用大O推导,得出最终的时间复杂度:

由于当i=0时,内循环执行了n次,当i = 1时,执行了n-1次,……当i=n-1时,执行了1次。所以总的执行次数为:

java圈复杂度工具idea 代码复杂度_时间复杂度

用我们推导大O阶的方法,第一条,没有加法常数不予考虑;第二条,只保留最高阶项,因此保留时(n^2)/2; 第三条,去除这个项相乘的常数,也就是去除1/2,最终这段代码的时间复杂度为O(n2)。

立方阶:

int i, j; 

for(i = 1; i < n; i++)

for(j = 1; j < n; j++)

for(j = 1; j < n; j++){

/*时间复杂度为O(1)的程序步骤序列*/

}//时间复杂度是O(n^3)

4,常见的时间复杂度

常见的时问复杂度如表所示:

java圈复杂度工具idea 代码复杂度_java圈复杂度工具idea_02

常用的时间复杂度所耗费的时间从小到大依次是:

java圈复杂度工具idea 代码复杂度_java圈复杂度工具idea_03

5,最坏情况和平均情况

最坏情况运行时间是一种保证,那就是运行时间将不会再坏了。 在应用中,这是一种最重要的需求, 通常, 除非特别指定, 我们提到的运行时间都是最坏情况的运行时间。

平均运行时间是所有情况中最有意义的,因为它是期望的运行时间。

注意:一般在没有特殊说明的情况下,都是指最坏时间复杂度。

二,空间复杂度

算法的空间复杂度通过计算算法所需的存储空间实现,算法空间复杂度的计算公式记作:S(n)= O(f(n)),其中,n为问题的规模,f(n)为语句关于n所占存储空间的函数。

一般情况下,一个程序在机器上执行时,除了需要存储程序本身的指令、常数、变量和输入数据外,还需要存储对数据操作的存储单元,若输入数据所占空间只取决于问题本身,和算法无关,这样只需要分析该算法在实现时所需的辅助单元即可。若算法执行时所需的辅助空间相对于输入数据量而言是个常数,则称此算法为原地工作,空间复杂度为0(1)。//即算法计算的辅助空间是已经确定分配好的固定空间。

三,时间和空间复杂度计算规则

1,加法规则
     T(n,m) = T1(n) + T2(m) = O(max{f(n), g(m)})//两个时间复杂度相加,取最大者为和
2,乘法规则
     T(n,m) = T1(n) * T2(m) = O(max{f(n)*g(m)})//
3,一个经验
     复杂度与时间效率的关系:
    c(常数)     <     logn     <     n     <     n*logn     <  n^2  <  n^3  < 2^n  <  3^n     <     n!
    l------------------------------l--------------------------l--------------l
                   较好                                                 一般                                          较差

四,常用算法的时间复杂度和空间复杂度

java圈复杂度工具idea 代码复杂度_运行时间_04