【落羽的落羽 C语言篇】动态内存管理·下

文章目录

  • 一、常见动态内存使用错误
      1. 对NULL空指针进行了解引用操作
      1. 对动态开辟的空间越界访问
      1. 对非动态内存开辟内存的空间使用free释放
      1. 使用free释放一块动态开辟内存的一部分
      1. 对一块内存进行多次free释放
      1. 动态开辟的内存没有free释放
  • 二、经典笔试题练习:动态内存管理
    • 练习1
    • 练习2
    • 练习3
    • 练习4

一、常见动态内存使用错误

1. 对NULL空指针进行了解引用操作

c 复制代码
int* p = (int*)malloc(INT_MAX);
*p = 12;
free(p);

p的空间可能会开辟失败,p有可能是NULL空指针。众所周知,空指针是不能直接进行解引用操作的。不对p检查就直接解引用,就可能出问题。

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

c 复制代码
int* p = (int*)malloc(10*sizeof(int));
assert(p!=NULL);
for(int i=0 ; i<12 ; i++)
    *(p+i) = i;
free(p);

很好理解,开辟好的内存空间多大就是多大,再去越界访问就不礼貌了。

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

c 复制代码
int a = 6;
int* p = &a;
free(p);

上期讲过,free释放不是动态开辟的内存,结果是未知的,换句话说,是错误的。

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

c 复制代码
int* p = (int*)malloc(10);
assert(p != NULL);
p += 2;
free(p);

此时,p不再指向这块开辟空间的开头了。因为free回收的是以参数指针为起点的空间,所以p前的这两字节空间无法被free回收了。

5. 对一块内存进行多次free释放

c 复制代码
int* p = (int*)malloc(10);
assert(p != NULL);
free(p);
free(p);

重复free,无意义

6. 动态开辟的内存没有free释放

c 复制代码
int* p = (int*)malloc(10);
assert(p != NULL);
*p = 20;
......

忘记释放使用完的动态内存,会造成内存泄漏

牢记:malloc(或calloc)和free是成对出现的,动态开辟出的内存一定要释放,并且要正确释放。

二、经典笔试题练习:动态内存管理

以下的test函数都会产生什么效果?

练习1

c 复制代码
void GetMemory(char* p)
{
    p = (char*)malloc(100);
}

void test()
{
    char* str = NULL;
    GetMemory(str);
    strcpy(str, "Hello world");
    printf("%s",str);
}

解析:函数void GetMemory(char* p)传参str时,p是形式参数,是str的一份拷贝,在这个函数里面对p的申请空间不会影响str的空间。所以,离开GetMemory函数后str还是一个NULL空指针。而strcpy的完成中,其实是有一步解引用的操作的。对str解引用,就是非法访问了,程序会崩溃。

练习2

c 复制代码
char* GetMemory()
{
    char p[] = "Hello world";
    return p;
}

void test()
{
    char* str = NULL;
    str = GetMemory();
    printf("%s",str);
}

解析:函数char* GetMemory()虽然返回指针p,但p指向的空间会在退出函数的时候就被释放,返回的p变成了野指针,str也变成了野指针。打印出的结果是随机的:

练习3

c 复制代码
void GetMemory(char** p, int num)
{
    *p = (char*)malloc(num);
}

void test()
{
    char* str = NULL;
    GetMemory(&str, 100);
    strcpy(str, "Hello mihoyo");
    printf("%s", str);
}

从结果上看,这段代码是没有问题的:

唯一的问题是:使用malloc后没有free掉开辟的空间,会造成内存泄漏,在test函数的最后加一句free(str);就好了。

练习4

c 复制代码
void test()
{
    char* str = (char*)malloc(100);
    strcpy(str, "Hello mihoyo");
    free(str);
    if(str != NULL)
    {
        strcpy(str, "Hello mihoyo");
        printf("%s",str);
    }
}

解析:第一次的strcpy完成后,str指向的动态内存空间被free掉了。但str的值没有被改变,它还是指向着那一块空间,但那块空间已经不属于它了,str已经是一个野指针了。str并不是NULL空指针,所以还会进入下面的if语句中,但strcpy和printf函数的执行就是非法访问了。结果可能是对的,但是有很大的安全隐患。

本篇完,感谢阅读

相关推荐
LDR00640 分钟前
Type-C 快充全面升级!LDR6601 赋能个人护理便携电机,重塑剃须刀 / 理发器新体验
c语言·开发语言
Luminous.2 小时前
C语言--day30
c语言·开发语言
玖玥拾2 小时前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
謓泽2 小时前
C语言不是语法,是通往机器的地图。
c语言·开发语言
不会C语言的男孩3 小时前
Linux 系统编程 · 第 8 章:进程基础
linux·c语言
2601_951643884 小时前
C语言长文整理,关键字和数据类型
c语言·数据类型·关键字·嵌入式开发·格式化输出
m0_547486666 小时前
《C#语言程序设计与实践》 全套PPT课件
c语言·c#·c语言程序设计
✎ ﹏梦醒͜ღ҉繁华落℘7 小时前
编程基础 --高内聚,低耦合
c语言·单片机
QK_008 小时前
C语言 static 关键字三大作用
c语言·开发语言
隔窗听雨眠8 小时前
C语言函数递归从入门到精通(下):性能优化与工程实践
c语言·算法·性能优化