文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C语言动态内存管理malloc函数怎么使用

2023-07-04 12:25

关注

这篇文章主要讲解了“C语言动态内存管理malloc函数怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言动态内存管理malloc函数怎么使用”吧!

1.C语言动态内存管理库函数介绍

1.1为什么存在动态内存管理

我们已经掌握的内存开辟方式有:

int val = 20;//在栈空间上开辟四个字节char arr[10] = {0};//在栈空间上开辟10个字节的连续空间

但是上述的开辟空间的方式有两个特点:

但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。也就是说当我们在定义变量时并不知道会使用多少的内存,这时候就需要进行动态内存开辟! 上述两种开辟内存方法一个在栈上开辟,一个在堆上开辟。 在C语言<stdlib.h>或<malloc.h>内置的库中有能够进行动态内存开辟的库函数。

1.2动态内存管理函数

1.2.1malloc

//Allocates memory blocks.void *malloc( size_t size );

参数size_t size表示需要开辟的内存的字节数。该函数会返回开辟好内存的首地址,如果开辟失败返回NULL

比如使用malloc函数开辟拥有10个整型元素的数组,那需要开辟的字节数为40字节。

#include <stdio.h>#include <stdlib.h>int main(){//使用malloc开辟一个含10个的整型元素数组int* arr = NULL;int* p = (int*)malloc(sizeof(int) * 10);//为数组开辟内存if (p == NULL){printf("内存申请失败!\n");exit(-1);//内存申请失败,程序没有再进行的必要,直接强制结束程序}arr = p;//确认内存开辟成功再将此内存交给数组p = NULL;int i = 0;for (i = 0; i < 10; i++){arr[i] = i + 1;printf("%d ", arr[i]);}return 0;}

因为malloc函数的返回值类型为void*,所以需要将已经开辟好的内存的首地址强制转换成整型指针类型。

运行结果:

1 2 3 4 5 6 7 8 9 10
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 22188)已退出,代码为 0。
按任意键关闭此窗口. . .

1.2.2free

对于动态内存开辟的空间,开辟的地址是在堆上的,使用完了是需要返还给操作系统的,C语言中专门有一个回收动态开辟内存的函数&mdash;&mdash;free。当然,程序结束时,会自动释放内存。

//Deallocates or frees a memory block.void free( void *memblock );

参数void *memblock表示动态开辟内存的首地址,注意这个地址只能是动态开辟内存的首地址,其他的地址都不行!如果传入的地址为NULL,则这个函数什么都不会做。 在上面所举例创建10个整型数组的程序中,就忽略了动态内存的释放,存在内存泄漏的风险。所以正确完整的程序应该为:

#include <stdio.h>#include <stdlib.h>int main(){//使用malloc开辟一个含10个的整型元素数组int* arr = NULL;int* p = (int*)malloc(sizeof(int) * 10);//为数组开辟内存if (p == NULL){printf("内存申请失败!\n");exit(-1);//内存申请失败,程序没有再进行的必要,直接强制结束程序}arr = p;//确认内存开辟成功再将此内存交给数组p = NULL;int i = 0;for (i = 0; i < 10; i++){arr[i] = i + 1;printf("%d ", arr[i]);}free(arr);//有借有还,再借不难arr = NULL;//好习惯:内存释放后,将指针变量置空return 0;}

运行结果:

1 2 3 4 5 6 7 8 9 10
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 23232)已退出,代码为 0。
按任意键关闭此窗口. . .

内存泄漏的危害: 如果动态内存已经使用完了,但不还给操作系统,也就是没有释放内存,就有可能造成内存泄漏的风险。对于其危害,举个栗子,如果在服务器上存在内存泄漏,则可能造成服务器崩溃。因为服务器是一直工作的,一旦存在内存泄漏,使用完的内存不还回去,久而久之,服务器内存被占用的越来越多,终有一天由于内存不足而造成服务器崩溃。

1.2.3calloc

该函数功能与malloc非常相似,仅仅多了个初始化的功能,就是说在动态内存开辟时,自动将内存中的元素初始化为0

//Allocates an array in memory with elements initialized to 0.void *calloc( size_t num, size_t size );

参数size_t num表示元素个数,size_t size表示每个元素所占字节数大小。

int main(){//使用malloc开辟一个含10个的整型元素数组int* arr = NULL;int* p = (int*)calloc(10, sizeof(int));//为数组开辟内存if (p == NULL){printf("内存申请失败!\n");exit(-1);//内存申请失败,程序没有再进行的必要,直接强制结束程序}arr = p;//确认内存开辟成功再将此内存交给数组p = NULL;int i = 0;for (i = 0; i < 10; i++){printf("%d ", arr[i]);}free(arr);//有借有还,再借不难arr = NULL;//好习惯:内存释放后,将指针变量置空return 0;}

运行结果:

0 0 0 0 0 0 0 0 0 0
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 29776)已退出,代码为 0。
按任意键关闭此窗口. . .

1.2.4realloc

该函数能够在保留原数据的情况下,对动态申请内存的大小进行调整,通常用来对数组或者链表等数据结构进行扩容。该函数在调整动态内存大小时有以下两个细节:

如果调整失败,返回NULL,调整成功返回新申请内存的首地址。

//Reallocate memory blocks.void *realloc( void *memblock, size_t size );

参数 void *memblock表示需要调整空间的首地址(必须为动态开辟的内存空间),参数size_t size表示调整后内存的字节数。

将动态申请的整型数组元素个数调整至20。

int main(){//使用malloc开辟一个含10个的整型元素数组int* arr = NULL;int* p = (int*)calloc(10, sizeof(int));//为数组开辟内存if (p == NULL){printf("内存申请失败!\n");exit(-1);//内存申请失败,程序没有再进行的必要,直接强制结束程序}arr = p;//确认内存开辟成功再将此内存交给数组p = NULL;//增加数组元素个数为20p = (int*)realloc(arr, sizeof(int) * 20);if (p == NULL){printf("内存调整失败!\n");exit(-1);//内存调整失败,程序没有再进行的必要,直接强制结束程序}arr = p;p = NULL;int i = 0;for (i = 0; i < 20; i++){arr[i] = i + 1;printf("%d ", arr[i]);}free(arr);//有借有还,再借不难arr = NULL;//好习惯:内存释放后,将指针变量置空return 0;}

运行结果:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 7520)已退出,代码为 0。
按任意键关闭此窗口. . .

1.3动态内存管理函数易错点

1.3.1对NULL指针的解引用操作

错误示范:

void test(){int* p = (int*)malloc(INT_MAX / 4);*p = 20;//如果p的值是NULL,就会有问题free(p);}

改正:

void test(){int* p = (int*)malloc(INT_MAX / 4);if (p == NULL){printf("内存申请失败!\n");exit(-1);//强制结束程序}*p = 20;//如果p的值是NULL,就会有问题free(p);}

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

错误示范:

void test(){int i = 0;int* p = (int*)malloc(10 * sizeof(int));if (NULL == p){exit(EXIT_FAILURE);}for (i = 0; i <= 10; i++){*(p + i) = i;//当i是10的时候越界访问}free(p);}

改正:

void test(){int i = 0;int* p = (int*)malloc(10 * sizeof(int));if (NULL == p){exit(EXIT_FAILURE);}for (i = 0; i < 10; i++){*(p + i) = i;//当i是10的时候越界访问}free(p);}

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

错误示范:

void test(){int a = 10;int* p = &a;free(p);//对非动态开辟的内存释放是错误的,程序会崩溃}

改正:

void test(){int* a = (int*)malloc(sizeof(int));if (a == NULL){exit(-1);//强制结束程序}*a = 10;int* p = a;free(p);//对非动态开辟的内存释放是错误的,程序会崩溃p = NULL;a = NULL;}

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

错误示范:

void test(){int* p = (int*)malloc(100);p++;free(p);//p不再指向动态内存的起始位置,程序崩溃}

改正:

void test(){int* p = (int*)malloc(100);free(p);//p不再指向动态内存的起始位置,程序崩溃p = NULL;}

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

错误示范:

void test(){int* p = (int*)malloc(100);free(p);free(p);//重复释放,程序崩溃}

改正:

void test(){int* p = (int*)malloc(100);free(p);}

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

错误示范:

void test(){int* p = (int*)malloc(100);if (NULL != p){*p = 20;}}int main(){test();while (1);//内存忘记示范,内存泄漏,程序崩溃}

改正:

void test(){int* p = (int*)malloc(100);if (NULL != p){*p = 20;}free(p);p = NULL;//好习惯}int main(){test();while (1);}

2.C语言动态内存管理库函数应用

2.1常见相关笔试题

//1.Test运行结果是什么?void GetMemory(char* p) {p = (char*)malloc(100);}void Test(void) {char* str = NULL;GetMemory(str);strcpy(str, "hello world");printf(str);}//2.Test运行结果是什么?char* GetMemory(void) {char p[] = "hello world";return p;}void Test(void) {char* str = NULL;str = GetMemory();printf(str);}//3.Test运行结果是什么?void GetMemory(char** p, int num) {*p = (char*)malloc(num);}void Test(void) {char* str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);}//4.Test运行结果是什么?void Test(void) {char* str = (char*)malloc(100);strcpy(str, "hello");free(str);if (str != NULL){strcpy(str, "world");printf(str);}}

题1:函数GetMemory的形参为char* pp为该函数的局部变量,作用域在函数内部,出了函数该变量就被销毁了,并且没有对申请好的内存进行释放。所以参数str传入函数GetMemory后,其值不会改变,仍为NULL,空地址是不能被用户访问修改的,因此程序崩溃。

题2pGetMemory函数内部的局部变量,该函数运行完后,其栈帧被销毁,在函数外得到返回的地址并访问属于非法访问,打印该地址的字符串,如果该空间没有被覆盖,能够打印hello world,否则打印随机值。调用printf函数是有可能覆盖该地址的,所以极大概率打印的是随机值。

运行结果:

烫烫烫烫烫烫烫烫8
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 23592)已退出,代码为 0。
按任意键关闭此窗口. . .

题3:该程序虽然会输出hello,但是是存在内存泄漏的,因为最后并没有释放申请的内存。

hello
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 20096)已退出,代码为 0。
按任意键关闭此窗口. . .

应该改为:

void GetMemory(char** p, int num) {*p = (char*)malloc(num);}void Test(void) {char* str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);free(str);str = NULL;}

题4:输出world,将一个动态申请的空间释放,传入的指针变量是不会置空的,会成为一个野指针,所以我们要养成一个好习惯:释放一个空间,应将其传入的指针置空!

world
D:\gtee\C-learning-code-and-project\test_928\Debug\test_928.exe (进程 13356)已退出,代码为 0。
按任意键关闭此窗口. . .

2.2C/C++语言中的内存开辟

C语言动态内存管理malloc函数怎么使用

C/C++程序内存分配的几个区域:

实际上普通的局部变量是在栈区分配空间的,栈区的特点是在上面创建的变量出了作用域就销毁。 但是被static修饰的变量存放在数据段(静态区),数据段的特点是在上面创建的变量,直到程序结束才销毁 所以生命周期变长。

2.3柔性数组

2.3.1柔性数组特点与使用

也许你从来没有听说过柔性数组(flexible array)这个概念,但是它确实是存在的。 C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做『柔性数组』成员。

typedef struct st_type{int i;int a[0];//柔性数组成员}type_a;

有些编译器会报错无法编译可以改成:

typedef struct st_type{int i;int a[];//柔性数组成员}type_a;

柔性数组的特点:

typedef struct st_type{int i;int a[0];//柔性数组成员}type_a;int main(){printf("%d\n", sizeof(type_a));//输出的是4return 0;}

柔性数组的使用:

int main(){int i = 0;type_a* p = (type_a*)malloc(sizeof(type_a) + 100 * sizeof(int));//业务需求代码段p->i = 100;for (i = 0; i < 100; i++) {p->a[i] = i;}free(p);return 0;}

2.3.2柔性数组的优点

不使用柔性数组也是可以实现同样的需求的:

typedef struct st_type{int i;int* p_a;}type_a;int main(){type_a* p = (type_a*)malloc(sizeof(type_a));p->i = 100;p->p_a = (int*)malloc(p->i * sizeof(int));//业务需求代码段int i = 0;for (i = 0; i < 100; i++) {p->p_a[i] = i;}//释放空间free(p->p_a);p->p_a = NULL;free(p);p = NULL;return 0;}

但是使用柔性数组的优势是其空间使用是连续的,而不使用柔性数组,其空间使用相比于柔性数组是散落的。

柔性数组的优点:

第一个好处是:方便内存释放如果我们的代码是在一个给别人用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,所以你不能指望用户来发现这个事。所以,如果我们把结构体的内存以及其成员要的内存一次性分配好了,并返回给用户一个结构体指针,用户做一次free就可以把所有的内存也给释放掉。
第二个好处是:这样有利于访问速度. 连续的内存有益于提高访问速度,也有益于减少内存碎片。(提升不是特别明显)

感谢各位的阅读,以上就是“C语言动态内存管理malloc函数怎么使用”的内容了,经过本文的学习后,相信大家对C语言动态内存管理malloc函数怎么使用这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯