【C语言】:总结动态内存的常见错误

简介: 【C语言】:总结动态内存的常见错误

动态内存的常见错误


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

错误代码如下

#include <stdlib.h>
#include <string.h>
#include <errno.h>
int main()
{
  int* p = (int*)malloc(40);
  *p = 20;
  free(p);
  p = NULL;
  return 0;
}

原因是这里开辟空间后并没有对p进行检查判断,若此时p=NULL,则程序会崩溃。

正确代码如下:

#include <stdlib.h>
#include <string.h>
#include <errno.h>
int main()
{
  int* p = (int*)malloc(40);
if (p == NULL)
{
  printf("%s\n", strerror(errno));
  return 1;
}
  *p = 20;
  free(p);
  p = NULL;
  return 0;
}

2.对动态开辟空间的越界访问

错误代码如下:

int main()
{
  int* p = (int*)malloc(40);
  if (p == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
   //使用
  int i = 0;
  for (i = 0; i <= 10; i++)
  {
    p[i] = i;
  }
  free(p);
  p = NULL;
  return 0;
}

原因只开辟了40个字节的空间,但在使用时用了44个,形成了越界访问

3.对非动态开辟内存使用free释放

错误代码如下:

int main()
{
  int a = 10;
  int* p = &a;
  //.....
  free(p);
  p = NULL;
  return 0;
}

这是典型的错误,free只能用来释放malloc,calloc,realloc动态开辟在堆区的空间,而上述代码里的变量a储存在栈区,无法使用free释放。

4.使用free释放一块动态开辟内存的一部分

错误代码如下:

int main()
{
  int* p = (int*)malloc(40);
  if (p == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  //使用
  int i = 0;
  for (i = 0; i < 6; i++)
  {
    *p = i;
    p++;
  }
  //释放
  free(p);//此时p不在起始位置,无法释放该块空间
  p = NULL;
  return 0;
}

这也是十分典型的错误,在for循环中p++后,p的地址已经改变了,此时p不在这块空间起始位置,一定是不能释放这块空间的!

正确代码为:

int main()
{
  int* p = (int*)malloc(40);
  if (p == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  //使用
  int i = 0;
  for (i = 0; i < 6; i++)
  {
    *(p+i) = i;
  }
  //释放
  free(p);//此时p不在起始位置,无法释放该块空间
  p = NULL;
  return 0;
}

5.对同一块动态内存多次释放

错误代码如下:

int main()
{
  int* p = (int*)malloc(40);
  if (p == NULL)
  {
    printf("%s\n", strerror(errno));
    return 1;
  }
  //...
  free(p);
  //...   //修改:只释放一次或是p=NULL
  free(p);
  return 0;
}

原因是第一个free时p所指向的空间已经还给系统了,但是p中存的还是刚刚开辟那块空间的起始地址,p是个野指针,执行到第二个free时会有问题。

6.动态开辟内存忘记释放(内存泄漏)

错误代码如下:

void test()
{
  int* p = (int*)malloc(100);
  //....
  int flag = 0;
  scanf("%d", &flag);
  if (flag == 5)
  {
    return 0;
  }
  free(p);
  p = NULL;
}
int main()
{
  test();
  //...
  return 0;
}

原因是,当我们调用test函数时,输入5,该函数直接结束,开辟的空间没有释放,这时会造成内存泄漏。

所以我们在使用动态内存函数时一定要注意。

目录
相关文章
|
1月前
|
存储 编译器 C语言
【C语言篇】数据在内存中的存储(超详细)
浮点数就采⽤下⾯的规则表⽰,即指数E的真实值加上127(或1023),再将有效数字M去掉整数部分的1。
|
24天前
|
存储 大数据 C语言
C语言 内存管理
本文详细介绍了内存管理和相关操作函数。首先讲解了进程与程序的区别及进程空间的概念,接着深入探讨了栈内存和堆内存的特点、大小及其管理方法。在堆内存部分,具体分析了 `malloc()`、`calloc()`、`realloc()` 和 `free()` 等函数的功能和用法。最后介绍了 `memcpy`、`memmove`、`memcmp`、`memchr` 和 `memset` 等内存操作函数,并提供了示例代码。通过这些内容,读者可以全面了解内存管理的基本原理和实践技巧。
|
24天前
|
缓存 Linux C语言
C语言 多进程编程(六)共享内存
本文介绍了Linux系统下的多进程通信机制——共享内存的使用方法。首先详细讲解了如何通过`shmget()`函数创建共享内存,并提供了示例代码。接着介绍了如何利用`shmctl()`函数删除共享内存。随后,文章解释了共享内存映射的概念及其实现方法,包括使用`shmat()`函数进行映射以及使用`shmdt()`函数解除映射,并给出了相应的示例代码。最后,展示了如何在共享内存中读写数据的具体操作流程。
|
1月前
|
存储 程序员 C语言
【C语言】动态内存管理
【C语言】动态内存管理
|
1月前
|
存储 编译器 C语言
C++内存管理(区别C语言)深度对比
C++内存管理(区别C语言)深度对比
67 5
|
1月前
|
C语言
C语言动态内存管理
C语言动态内存管理
30 4
|
1月前
|
存储 NoSQL 程序员
C语言中的内存布局
C语言中的内存布局
35 0
|
1月前
|
C语言
【C语言篇】字符和字符串以及内存函数详细介绍与模拟实现(下篇)
perror函数打印完参数部分的字符串后,再打印⼀个冒号和⼀个空格,再打印错误信息。
|
1月前
|
存储 安全 编译器
【C语言篇】字符和字符串以及内存函数的详细介绍与模拟实现(上篇)
当然可以用scanf和printf输入输出,这里在之前【C语言篇】scanf和printf万字超详细介绍(基本加拓展用法)已经讲过了,这里就不再赘述,主要介绍只针对字符的函数.
|
1月前
|
C语言 索引
C语言编译环境中的 调试功能及常见错误提示
这篇文章介绍了C语言编译环境中的调试功能,包括快捷键操作、块操作、查找替换等,并详细分析了编译中常见的错误类型及其解决方法,同时提供了常见错误信息的索引供参考。