C语言基础知识——动态内存管理

目录

学习目标:

学习内容:

一、为什么存在动态内存分配

二、动态内存函数的介绍

         2.1  malloc

        2.2  free

        2.3  calloc

        2.4 realloc

三、常见的动态内存错误

四、几个经典的笔试题

五、柔性数组

学习时间:

学习产出:


学习目标:

  • 学习动态内存管理


学习内容:

一、为什么存在动态内存分配

我们知道数组的创建之后,其大小也就确定了。

如:int arr[20] = {0};

1. 空间开辟大小是固定的。

2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。

因此有时候我们的数组又不能被固定大小,这时候就需要动态开辟空间

二、动态内存函数的介绍

         2.1  malloc

 返回类型是指针类型,括号中是申请开辟的空间的大小

这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。

        如果开辟成功,则返回一个指向开辟好空间的指针。

        如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。

        返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己 来决定。

        如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。

        2.2  free

释放掉开辟的申请的空间

要注意的几个点:

        参数为NULL时什么都不做

        参数不能是非动态的,且free函数的行为是未定义的。 

        2.3  calloc

        这个是malloc的升级,开辟num个,大小为size的空间,并且把空间都初识化为0

        与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。 

        2.4 realloc

 当我们申请的空间不够用时,可以用realloc来继续申请;

        realloc函数的出现让动态内存管理更加灵活。

         有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时 候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小 的调整。

         ptr 是要调整的内存地址

         size 调整之后新大小

         返回值为调整之后的内存起始位置。

         这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到 新 的空间。

        realloc在调整内存空间的是存在两种情况:

                 情况1:原有空间之后有足够大的空间

                 情况2:原有空间后面没有足够连续空间开辟

                                则函数会在找另一个足够的地方,开辟空间。先将原有的数据拷贝一份到新位置上。原来位置的销毁。

        情况1 当是情况1 的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化。

        情况2 当是情况2 的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适大小 的连续空间来使用。这样函数返回的是一个新的内存地址。 由于上述的两种情况,realloc函数的使用就要注意一些。

三、常见的动态内存错误

3.1对NULL指针解引用

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

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

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

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

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

3.6忘记释放申请开辟的动态内存

这可能会导致内存泄漏,很严重的事情 

五、柔性数组

C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做『柔性数组』成员。

柔性数组的特点:

          结构中的柔性数组成员前面必须至少一个其他成员。

          sizeof 返回的这种结构大小不包括柔性数组的内存。

         包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大 小,以适应柔性数组的预期大小。

 柔性数组的使用:

struct S {
	int i;
	int arr[];
};

int main()
{
	struct S s;
	struct S* ptr = (struct S*)malloc(sizeof(struct S) + 100 * sizeof(int));
	if (ptr == NULL)
	{
		return;
	}
	ptr->i = 100;
	int i = 0;
	for (i = 0; i < 100; i++)
	{
		ptr->arr[i] = i;
	}
	for (i = 0;i < 100; i++)
	{
		printf("%d ", ptr->arr[i]);
	}

	free(ptr);
	ptr = NULL;

	return 0;
}

使用柔性数组的好处:

        方便内存的释放

        提高访问速度


学习时间:

二零二二年十月


学习产出:

  • CSDN 技术博客 1 篇
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Though even

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值