C语言中动态内存管理初学者容易犯的错误有哪些

其他教程   发布日期:2025年03月15日   浏览次数:226

这篇“C语言中动态内存管理初学者容易犯的错误有哪些”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C语言中动态内存管理初学者容易犯的错误有哪些”文章吧。

1.对NULL指针的解引用操作

当你malloc一块空间时,是有可能开辟失败的。一旦失败,malloc会返回NULL指针。如果不判断malloc的返回值,就直接使用malloc返回的指针,有可能导致对NULL指针的解引用操作。例如:

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. for (int i = 0; i < 10; i++)
  3. {
  4. p[i] = i + 1;
  5. }

由于malloc可能返回NULL指针,一旦p为NULL,对其解引用是非常危险的!

解决方法:每次使用malloc, calloc, realloc等函数时,一定要判断返回值是否为NULL。

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. // 错误处理
  5. perror("malloc");
  6. exit(-1);
  7. }
  8. // 此时p一定不为NULL
  9. for (int i = 0; i < 10; i++)
  10. {
  11. p[i] = i + 1;
  12. }

2.对动态内存的越界访问

假设你使用malloc申请了40个字节的空间,一定要记住这块空间的大小,不能越界访问。比如:

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. // 错误处理
  5. perror("malloc");
  6. exit(-1);
  7. }
  8. for (int i = 0; i <= 10; i++)
  9. {
  10. p[i] = i + 1;
  11. }

上面的代码中,当i==10时,p[i]等价于*(p+10),已经超出了申请的40个字节的空间,这是非常危险的!

解决方法:无论如何,在用指针访问内存时,一定要观察其是否越界!比如上面的代码,只能访问p[0]~p[9]的空间。

3.忘记free

如果你使用malloc等函数,开辟了空间,但是确没有free,就会导致内存泄漏!对于程序员来说,这种情况是非常不舒服的,一定要避免。

解决方法:对于每一块动态内存开辟的空间,使用完后都要free。比如:

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. perror("malloc");
  5. exit(-1);
  6. }
  7. // 使用
  8. // ...
  9. free(p);
  10. p = NULL;

4.对同一块空间free两次

如果已经free过一块空间了,如果再free一次,那不多此一举吗!不仅如此,如果第一次free之后没有把这个指针置为NULL,第二次free的时候,这个指针是一个野指针,是非常危险的!

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. perror("malloc");
  5. exit(-1);
  6. }
  7. // 使用
  8. // ...
  9. free(p);
  10. // ...
  11. free(p);

解决方法:你free一次就行了,干啥要free两次呀!

值得说明的是,由于free(NULL);时,free函数啥也不干,所以下面的代码是不会有问题的,但也不建议这么写(有点吃饱了撑着的感觉)。

  1. // ...
  2. free(p);
  3. p = NULL;
  4. free(p);

5.free动态内存的一部分

注意,free一个指针的时候,这个指针必须指向动态内存的起始位置!比如下面的代码中,如果这个指针已经不指向起始位置了,free的时候也是非常危险的。

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. perror("malloc");
  5. exit(-1);
  6. }
  7. // ...
  8. p += 5;
  9. // ...
  10. free(p);
  11. p = NULL;

由于在malloc之后,free之前,指针p已经不指向动态内存的起始位置了,此时再free掉p就会出问题。

解决方法:由于free的时候需要知道动态内存的起始地址,在使用这块内存的时候不建议改变p的指向。除此之外,如果使用realloc扩容,一定要更新p,时刻让p指向动态内存的起始地址,方便free释放。

  1. int* p = (int*)malloc(10 * sizeof(int));
  2. if (p == NULL)
  3. {
  4. perror("malloc");
  5. exit(-1);
  6. }
  7. // ...
  8. // 使用时时刻让p指向动态内存的起始位置!
  9. int* tmp = (int*)realloc(p, 20 * sizeof(int));
  10. if (tmp == NULL)
  11. {
  12. free(p);
  13. p = NULL;
  14. perror("realloc");
  15. exit(-1);
  16. }
  17. else
  18. {
  19. p = tmp;
  20. }
  21. // ...
  22. free(p);
  23. p = NULL;

6.free非动态开辟的内存

free是用来释放动态开辟的内存的。所以,如果这块内存空间不是用malloc, calloc, realloc动态开辟的,就别free啦。

  1. int arr[10] = {0};
  2. // ...
  3. int a = 0;
  4. int* pa = &a;
  5. // ...
  6. free(arr);
  7. arr = NULL;
  8. free(pa);
  9. pa = NULL;

解决方法:不要写出这样的代码,free的时候得看清楚,这块空间到底是不是动态开辟的空间,时刻保持头脑清醒。

以上就是C语言中动态内存管理初学者容易犯的错误有哪些的详细内容,更多关于C语言中动态内存管理初学者容易犯的错误有哪些的资料请关注九品源码其它相关文章!