广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言动态内存管理malloc柔性数组示例详解
  • 306
分享到

C语言动态内存管理malloc柔性数组示例详解

C语言动态内存malloc柔性数组C语言 malloc柔性数组 2022-11-13 19:11:17 306人浏览 独家记忆
摘要

目录1.C语言动态内存管理库函数介绍?1.1为什么存在动态内存管理?1.2动态内存管理函数?1.2.1malloc?1.2.2free?1.2.3calloc?1.2.4reallo

大家好!在实现动态通讯录的时候,我用到了malloc 和realloc动态申请内存,现在我们就来好好聊一聊动态内存管理。

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语言中专门有一个回收动态开辟内存的函数——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;
	//增加数组元素个数为20
	p = (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/C++程序内存分配的几个区域:

  • 栈区(stack):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。
  • 堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配方式类似于链表。
  • 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
  • 代码段:存放函数体(类成员函数和全局函数)的二进制代码。

实际上普通的局部变量是在栈区分配空间的,栈区的特点是在上面创建的变量出了作用域就销毁。 但是被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;

柔性数组的特点:

  • 结构中的柔性数组成员前面必须至少一个其他成员。
  • sizeof 返回的这种结构大小不包括柔性数组的内存。
  • 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。
typedef struct st_type
{
	int i;
	int a[0];//柔性数组成员
}type_a;
int main()
{
	printf("%d\n", sizeof(type_a));//输出的是4
	return 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柔性数组的资料请关注编程网其它相关文章!

--结束END--

本文标题: C语言动态内存管理malloc柔性数组示例详解

本文链接: https://www.lsjlt.com/news/170367.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C语言动态内存管理malloc柔性数组示例详解
    目录1.C语言动态内存管理库函数介绍1.1为什么存在动态内存管理1.2动态内存管理函数1.2.1malloc1.2.2free1.2.3calloc1.2.4realloc1.3动态...
    99+
    2022-11-13
    C语言动态内存malloc柔性数组 C语言 malloc柔性数组
  • 详解C语言中动态内存管理及柔性数组的使用
    目录一、malloc二、free三、calloc四、realloc1、realloc在扩容时的情况2、realloc也能实现malloc功能五、使用动态内存的常见错误1、free空指...
    99+
    2022-11-13
  • C语言编程之动态内存与柔性数组的了解
    目录介绍动态内存分配函数常见的动态内存开辟的错误举个例子柔性数组柔性数组的特点创建变量 1,局部变量–栈区 2,全局变量–静态区 创建数组 1,局部数组–栈区 2,全局数组–静态区 ...
    99+
    2022-11-12
  • C语言动态内存管理malloc函数怎么使用
    这篇文章主要讲解了“C语言动态内存管理malloc函数怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言动态内存管理malloc函数怎么使用”吧!1.C语言动态内存管理库函数介绍1...
    99+
    2023-07-04
  • 详解C语言之动态内存管理
    目录开辟动态内存的函数释放开辟的动态内存空间的函数错误信息函数具体使用例: 常见的动态内存错误总结先来了解一下动态管理内存所需用到的函数 开辟动态内存的函数 1.mallo...
    99+
    2022-11-12
  • C语言编程C++柔性数组结构示例讲解
    目录绕指柔—柔性数组柔性数组的特点:第一个好处是:方便内存释放第二个好处是:这样有利于访问速度总结绕指柔—柔性数组 也许你从来没有听说过柔性数组(flexible array)这个概...
    99+
    2022-11-12
  • 详解C语言中的动态内存管理
    目录一、动态内存管理1.1为什么要有动态内存管理1.2动态内存介绍1.3常见的动态内存错误一、动态内存管理 1.1为什么要有动态内存管理 1.1.1  在c语言中我们普通的...
    99+
    2022-12-12
    C语言动态内存管理 C语言 内存管理 C语言 内存
  • C语言中动态内存管理的示例分析
    这篇文章主要介绍了C语言中动态内存管理的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。什么是动态内存分配我们都知道在C语言中,定义变量的时候,系统就会为这个变量分配内...
    99+
    2023-06-25
  • C语言中动态内存管理图文详解
    目录1.动态内存开辟的原因2.动态内存函数的介绍2.1malloc和free2.2calloc2.3realloc3.常见的动态内存错误3.1对NULL指针的解引用操作3.2对动态开...
    99+
    2022-11-13
  • C语言动态内存函数详解
    目录动态开辟空间的原因 1、malloc函数2、free函数3、calloc函数4、realloc函数总结动态开辟空间的原因 静态开辟空间是固定的,数组在申明的时候必须指定数组的长...
    99+
    2022-11-12
  • C语言中动态内存分配malloc、calloc和realloc函数解析
    目录前言free函数malloc函数calloc函数realloc函数扩充malloc/calloc/realloc区别总结总结前言 有时候我们需要的空间大小不确定,需要随着程序需要...
    99+
    2022-11-13
  • C语言中动态内存管理实例分析
    今天小编给大家分享一下C语言中动态内存管理实例分析的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1.动态内存开辟的原因常见的...
    99+
    2023-07-02
  • C语言动态内存的分配实例详解
    目录前言动态内存分配的定义动态内存的优势<1>  可以控制内存的大小<2> 可以多次利用这部分空间<3>不占用栈区的内存malloc c...
    99+
    2022-11-13
  • C语言 动态内存管理全面解析
    目录1. 为什么存在动态内存分配2. 动态内存函数的介绍2.1 malloc和free2.2 calloc 2.3 realloc3. 常见的动态内存错误3.1 对NULL...
    99+
    2022-11-13
  • C语言编程C++动态内存分配示例讲解
    目录动态内存管理为什么存在动态内存分配动态内存函数的介绍malloc申请空间和free释放空间有借有还 free释放内存calloc申请内存realloc调整动态内存的大小reall...
    99+
    2022-11-12
  • C语言 动态分配数组案例详解
    目录一维动态数组的创建:二维数组的创建:很多人在编写C语言代码的时候很少使用动态数组,不管什么情况下通通使用静态数组的方法来解决,在当初学习C语言的时候我就是一个典型的例子,但是现在...
    99+
    2022-11-12
  • C语言内存管理及初始化细节示例详解
    目录地址空间指针与内存关系内存分配与初始化细节内存泄漏Cookie地址空间 首先我们回味一下之前的老图,这个图由于是我手残加 ppt 即时创作,又因为是C语言入门时讲的,内容非常粗...
    99+
    2022-11-13
  • C语言动态内存管理实例代码分析
    这篇文章主要介绍了C语言动态内存管理实例代码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言动态内存管理实例代码分析文章都会有所收获,下面我们一起来看看吧。1.动态内存开辟的原因常见的内存开辟方式int...
    99+
    2023-07-02
  • C语言 超详细梳理总结动态内存管理
    目录一.为什么存在动态内存分配二.动态内存函数的介绍1.malloc和free2.calloc3.realloc三.常见的动态内存错误1.对NULL指针的解引用操作2.对动态开辟空间...
    99+
    2022-11-13
  • 深入了解C语言的动态内存管理
    目录一、为什么会存在动态内存二、动态内存函数1.malloc和free2.calloc3.realloc三、动态内存函数常见错误2.对NULL指针进行解引用操作3.使用free释放一...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作