1.算法效率
1.1 如何衡量一个算法的好坏
long long Fib(int N)
{
if(N < 3)
return 1;
return Fib(N-1) + Fib(N-2);
}
当计算斐波那契数列时,使用递归的方法会导致计算量呈指数级增长,效率较低。这是因为在递归计算中会重复计算相同的子问题,造成了资源的浪费。
举个例子,我们来计算斐波那契数列的第n项,假设n=5。斐波那契数列的递归定义为:
如果我们采用递归的方式计算F(5),会涉及到很多重复计算,导致效率低下。示例代码如下:
#include <stdio.h>
int fibonacci_recursive(int n) {
if (n == 0) {
return 0;
} else if (n == 1) {
return 1;
} else {
return fibonacci_recursive(n-1) + fibonacci_recursive(n-2);
}
}
int main() {
int n = 5;
int result = fibonacci_recursive(n);
printf("The %dth Fibonacci number is: %d\n", n, result);
return 0;
}
在这个例子中,计算F(5)时会重复计算F(3)、F(2)等子问题,导致效率低下。因此,为了提高效率,我们可以考虑使用其他方法,如动态规划或迭代方法来计算斐波那契数列。
如果是算阶乘
#include <stdio.h>
int factorial(int n) {
if (n == 0) {
return 1;
} else {
return n * factorial(n - 1);
}
}
int main() {
int n;
printf("请输入一个整数 n:");
scanf("%d", &n);
if (n < 0) {
printf("输入的数必须为非负整数。\n");
} else {
int result = factorial(n);
printf("%d 的阶乘是 %d\n", n, result);
}
return 0;
}
我们把n的值加大到10的时候那么我们需要算10!=9* 8! 然后8!=8*7,一直都会有这种子问题下去这样会非常的浪费空间稍微大一点就会崩掉,如果采用迭代的方式来算n!,直接循环n次累乘很快变可以出现答案
1.2 算法的复杂度
1.3 复杂度在校招中的考察
2.时间复杂度
2.1 时间复杂度的概念
// 请计算一下Func1中++count语句总共执行了多少次?
void Func1(int N)
{
int count = 0;
for (int i = 0; i < N ; ++ i)
{
for (int j = 0; j < N ; ++ j)
{
++count;
}
}
for (int k = 0; k < 2 * N ; ++ k)
{
++count;
}
int M = 10;
while (M--)
{
++count;
}
printf("%d\n", count);
}
- N = 10 F(N) = 130
- N = 100 F(N) = 10210
- N = 1000 F(N) = 1002010
在很多大学竞赛中很多算法都采用时间复杂度来区分一个选手的得分
例如下面是我们学校的一道竞赛题
一般会用很多数据去测你写的程序如果时间复杂度不达标就算写出了效果也拿不到满分
就那这题举例如果我们写了一个算斐波那契的函数然后再去双重循环累加,那么恭喜你喜提百分之20的巨额分数
满分代码:
2.2 大O的渐进表示法
2.3常见时间复杂度计算举例
// 计算Func2的时间复杂度?
void Func2(int N)
{
int count = 0;
for (int k = 0; k < 2 * N ; ++ k)
{
++count;
}
int M = 10;
while (M--)
{
++count;
}
printf("%d\n", count);
}
// 计算Func3的时间复杂度?
void Func3(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);
}
// 计算Func4的时间复杂度?
void Func4(int N)
{
int count = 0;
for (int k = 0; k < 100; ++ k)
{
++count;
}
printf("%d\n", count);
}
// 计算strchr的时间复杂度?
const char * strchr ( const char * str, int character );
实例5:
// 计算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;
}
}
// 计算BinarySearch的时间复杂度?
int BinarySearch(int* a, int n, int x)
{
assert(a);
int begin = 0;
int end = n-1;
// [begin, end]:begin和end是左闭右闭区间,因此有=号
while (begin <= end)
{
int mid = begin + ((end-begin)>>1);
if (a[mid] < x)
begin = mid+1;
else if (a[mid] > x)
end = mid-1;
else
return mid;
}
return -1;
}
// 计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{
if(0 == N)
return 1;
return Fac(N-1)*N;
}
// 计算斐波那契递归Fib的时间复杂度?
long long Fib(size_t N)
{
if(N < 3)
return 1;
return Fib(N-1) + Fib(N-2);
}
3.空间复杂度
// 计算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;
}
}
// 计算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;
}
// 计算阶乘递归Fac的空间复杂度?
long long Fac(size_t N)
{
if(N == 0)
return 1;
return Fac(N-1)*N;
}
4. 常见复杂度对比
5. 复杂度的oj练习
思路1:把1-n的数累加到sum里面然后再遍历这个数组,用sum减去数组的每一个数最后就会剩下那个缺少的数
int missingNumber(int* nums, int numsSize){
int sum=0;
for(int i=0;i<=numsSize;i++)
{
sum=sum+i;
}
for(int i=0;i<numsSize;i++)
{
sum=sum-nums[i];
}
return sum;
}
但是这种方式需要考虑一种极端的情况就是sum值溢出的问题
思路2:这里需要理解一个概念 x^0=x x^x=0,这里先让x去和0-n的数^一变,因为相同异或的数已经是0了那么剩下的x的值就是缺少的那个值,因为成对出现的都会是0,这里把x定义为0,0^x=0
所以这里好观察那个单独的数是什么
int missingNumber(int* nums, int numsSize){
int n=numsSize;
int x=0;
for(int i=0;i<=n;i++)
{
x=x^i;
}
for(int i=0;i<numsSize;i++)
{
x=x^nums[i];
}
return x;
}
第一种时间复杂度是o(n^2)所以这道题就不考虑了,这道题时间过不去
思路2代码实现 :
三段逆置法
void Reverse(int* array,int left, int right)
{
while(left<right)
{
int temp=array[left];
array[left]=array[right];
array[right]=temp;
left++;
right--;
}
}
void rotate(int* nums, int numsSize, int k) {
k=k%numsSize;
Reverse(nums,0,numsSize-k-1);
Reverse(nums,numsSize-k,numsSize-1);
Reverse(nums,0,numsSize-1);
}
这个时间复杂度是o(n)就可以过
思路3:
思路2如果提前没有做过,第一次做,基本不可能能直接现场推出来,一般人只能学习大佬留下来的方法
void rotate(int* nums, int numsSize, int k) {
k=k%numsSize;
int temp[numsSize];
int n=numsSize;
memcpy(temp,nums+n-k,sizeof(int)*k);
memcpy(temp+k,nums,sizeof(int)*(n-k));
memcpy(nums,temp,sizeof(int)*n);
}