详解复杂度的计算

详解复杂度的计算

文章目录

算法的时间复杂度和空间复杂度⭐️1.算法效率⭐️2.时间复杂度1.1时间复杂度的概念1.2大O的渐进表示法1.3常见例题

⭐️3.空间复杂度⭐️4. 常见复杂度对比

算法的时间复杂度和空间复杂度

⭐️1.算法效率

算法的复杂度:

1.算法在编写成可执行程序后,运行 时需要耗费时间资源和空间(内存)资源 。因此衡量一个算法的好坏,一般是从时间和空间两个维度来衡量的,即时间复杂度和空间复杂度。 2.时间复杂度主要衡量一个算法的运行快慢,而空间复杂度主要衡量一个算法运行所需要的额外空间。在计算机发展的早期,计算机的存储容量很小。所以对空间复杂度很是在乎。但是经过计算机行业的迅速发展,计算机的存储容量已经达到了很高的程度。所以我们如今已经不需要再特别关注一个算法的空间复杂度。

⭐️2.时间复杂度

1.1时间复杂度的概念

时间复杂度的定义:

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

实际中我们计算时间复杂度时,我们其实并不一定要计算精确的执行次数,而只需要大概执行次数,那么这里我们使用大O的渐进表示法。

1.2大O的渐进表示法

大O符号(Big O notation):是用于描述函数渐进行为的数学符号。 推导大O阶方法: 1、用常数1取代运行时间中的所有加法常数。 2、在修改后的运行次数函数中,只保留最高阶项。 3、如果最高阶项存在且不是1,则去除与这个项目相乘的常数。得到的结果就是大O阶。

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

1.3常见例题

void Func1(int N)

{

int count = 0;

for (int k = 0; k < 2 * N; ++k)

{

++count;

}

int M = 10;

while (M--)

{

++count;

}

printf("%d\n", count);

}

基本操作执行了2N+10次,通过推导大O阶方法知道,时间复杂度为 O(N)

void Func2(int N, int M)

{

int count = 0;

for (int k = 0; k < M; ++k)

{

++count;

}

for (int k = 0; k < N; ++k)

{

++count;

}

printf("%d\n", count);

}

基本操作执行了M+N次,有两个未知数M和N,时间复杂度为 O(N+M)

void Func4(int N)

{

int count = 0;

for (int k = 0; k < 100; ++ k)

{

++count;

}

printf("%d\n", count);

}

基本操作执行了10次,通过推导大O阶方法,时间复杂度为 O(1)

// 计算strchr的时间复杂度?

const char * strchr ( const char * str, int character );

基本操作执行最好1次,最坏N次,时间复杂度一般看最坏,时间复杂度为 O(N)

// 计算BubbleSort的时间复杂度?

void BubbleSort(int* a, int n)

{

assert(a);

for (size_t end = n; end > 0; --end)

{

int exchange = 0;

for (size_t i = 1; i < end; ++i)

{

if (a[i - 1] > a[i])

{

Swap(&a[i - 1], &a[i]);

exchange = 1;

}

}

if (exchange == 0)

break;

}

}

基本操作执行最好N次,最坏执行了(N*(N+1)/2次,通过推导大O阶方法+时间复杂度一般看最 坏,时间复杂度为 O(N^2)

// 计算BinarySearch的时间复杂度?

int BinarySearch(int* a, int n, int x)

{

assert(a);

int begin = 0;

int end = n - 1;

while (begin < end)

{

int mid = begin + ((end - begin) >> 1);

if (a[mid] < x)

begin = mid + 1;

else if (a[mid] > x)

end = mid;

else

return mid;

}

return -1;

}

操作执行最好1次,最坏O(logN)次,时间复杂度为 O(logN) ps:logN在算法分析中表示是底数为2,对数为N。有些地方会写成lgN。(折纸查找计算出来)

// 计算阶乘递归Fac的时间复杂度?

long long Fac(size_t N)

{

if(0 == N)

return 1;

return Fac(N-1)*N;

}

基本操作递归了N次,时间复杂度为O(N)。

8

// 计算斐波那契递归Fib的时间复杂度?

long long Fib(size_t N)

{

if(N < 3)

return 1;

return Fib(N-1) + Fib(N-2);

}

基本操作递归了2N次,时间复杂度为O(2N)。

⭐️3.空间复杂度

空间复杂度也是一个数学表达式,是对一个算法在运行过程中临时占用存储空间大小的量度 。 空间复杂度不是程序占用了多少bytes的空间,因为这个也没太大意义,所以空间复杂度算的是变量的个数。空间复杂度计算规则基本跟实践复杂度类似,也使用大O渐进表示法。

注意:函数运行时所需要的栈空间(存储参数、局部变量、一些寄存器信息等)在编译期间已经确定好了,因此空间复杂度主要通过函数在运行时候显式申请的额外空间来确定。

// 计算BubbleSort的空间复杂度?

void BubbleSort(int* a, int n)

{

assert(a);

for (size_t end = n; end > 0; --end)

{

int exchange = 0;

for (size_t i = 1; i < end; ++i)

{

if (a[i - 1] > a[i])

{

Swap(&a[i - 1], &a[i]);

exchange = 1;

}

}

if (exchange == 0)

break;

}

}

使用了常数个额外空间,所以空间复杂度为 O(1)

// 计算Fibonacci的空间复杂度?

// 返回斐波那契数列的前n项

long long* Fibonacci(size_t n)

{

if (n == 0)

return NULL;

long long * fibArray = (long long *)malloc((n + 1) * sizeof(long long));

fibArray[0] = 0;

fibArray[1] = 1;

for (int i = 2; i <= n; ++i)

{

fibArray[i] = fibArray[i - 1] + fibArray[i - 2];

}

return fibArray;

}

动态开辟了N个空间,空间复杂度为 O(N)

// 计算阶乘递归Fac的空间复杂度?

long long Fac(size_t N)

{

if (N == 0)

return 1;

return Fac(N - 1)*N;

}

递归调用了N次,开辟了N个栈帧,每个栈帧使用了常数个空间。空间复杂度为O(N)

⭐️4. 常见复杂度对比

相关灵感

365365094 《部落冲突》如何改名字

《部落冲突》如何改名字

📅 09-25 👁️ 3626
365bet足球网开户 智能电视怎么看照片?巧用当贝市场,一键轻松上传观看
365365094 個人製作MAD影片的剪輯流程與剪輯心得。
365365094 日本海淘网站有哪些?5个热门日本海淘网站汇总攻略,海淘攻略
365bet足球网开户 Lisa (泰國歌手)

Lisa (泰國歌手)

📅 07-23 👁️ 5554
365bet足球网开户 雅鲁藏布大峡谷

雅鲁藏布大峡谷

📅 07-14 👁️ 9439
正规beat365旧版 《商君书》垦令 译文

《商君书》垦令 译文

📅 09-08 👁️ 5602
365365094 净水器水质检测标准大揭秘

净水器水质检测标准大揭秘

📅 06-30 👁️ 1929
365bet足球网开户 如何设置或解决忘记BIOS密码/UEFI密码/开机密码