动态内存分配

hello everybody!欢迎来到动态内存分配小课堂!接下来我将为大家讲解关于动态内存分配的一些小知识!

首先,我们为什么要使用内存分配?
我们知道,一般我们使用的内存开辟方式都是这样的:

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

但是上述的开辟空间的⽅式有两个特点:
• 空间开辟⼤⼩是固定的。
• 数组在申明的时候,必须指定数组的⻓度,数组空间⼀旦确定了⼤⼩不能调整
但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间⼤⼩在程序运⾏的时候才能知道,那数组的编译时开辟空间的⽅式就不能满⾜了。
C语⾔引⼊了动态内存开辟,让程序员⾃⼰可以申请和释放空间,就⽐较灵活了。

1. malloc

C语言提供了一个动态内存开辟的函数:

void* malloc(size_t size);

这个函数向内存申请⼀块连续可⽤的空间,并返回指向这块空间的指针。
• 如果开辟成功,则返回⼀个指向开辟好空间的指针。(自己创建一个指针来接受)
• 如果开辟失败,则返回⼀个 NULL 指针,因此malloc的返回值⼀定要做检查。(所以写程序在使用malloc之前,一定要对申请空间进行判断)
• 返回值的类型是 void * ,所以malloc函数并不知道开辟空间的类型,具体在使⽤的时候使⽤者⾃⼰来决定。(就是要记住强制转换使用的类型)
• 如果参数 size 为0,malloc的⾏为是标准是未定义的,取决于编译器。(尽量不要使用)

2. free

C语⾔提供了另外⼀个函数free,专⻔是⽤来做动态内存的释放和回收的,函数原型如下:

void free(void* ptr);

free函数⽤来释放动态开辟的内存。
• 如果参数 ptr 指向的空间不是动态开辟的,那free函数的⾏为是未定义的。(重点:ptr指向的空间一定要是动态开辟的空间,而不能是栈区的变量等等)
• 如果参数 ptr 是NULL指针,则函数什么事都不做。(虽然没用,但是不建议使用)
malloc和free都声明在 stdlib.h 头⽂件中。
可以看看这张图:

在这里插入图片描述
学习了这两个,当然也可以进行一定的尝试了,那么下面为大家展示一段代码:

#include<stdio.h>
#include<stdlib.h>//动态内存分配少了头文件可不行

int main()
{
	int* p = (int*)malloc(5 * sizeof(int));//用指针p接收malloc随机开辟的20个字节的空间

	if (p == NULL)//如果malloc开辟空间失败
	{
		perror("malloc");//不要忘了双引号
		return 1;
	}

	for (int i = 0; i < 5; i++)
	{
		*(p + i) = i+1;//给空间塞上值
	}

	for (int i = 0; i < 5; i++)
	{
		printf("%d ", p[i]);//打印
	}

	free(p);//用完之后记得释放
	p = NULL;//p要置为空指针,不然就成野指针了

	return 0;
}

大家一定注意到了一些注释标注的代码,而其中一些也是要讲解的重点:
①一开始的if语句,是为了检测malloc的内存分配是否成功,因为如果malloc内存分配失败就会返回NULL指针,而此时,我们也可以用perror函数来输出错误的原因。而返回值 1 通常被用作表示程序发生错误或异常情况的标志。在这里,返回 1 可以作为指示代码遇到内存分配失败的信号,并允许调用程序根据需要进行错误处理或采取其他适当的措施。
②用完指针p之后,后面就进行了free,使原本动态分配的内存被释放,防止内存泄漏,而之后将p指针置为NULL,也是为了防止其成为野指针造成内存泄漏

3. calloc

calloc作为C语言提供的一种动态内存分配的函数,原型也与malloc相似:

void* calloc(size_t num,size_t size); 

函数的功能是为 num 个⼤⼩为 size 的元素开辟⼀块空间,并且把空间的每个字节初始化为0。
• 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

#include<stdio.h>
#include<stdlib.h>

int main()
{
	int* p = (int*)calloc(5, sizeof(int));//calloc使用

	if (p == NULL)
	{
		perror("calloc");
		return 1;
	}

	for (int i = 0; i < 5; i++)
	{
		printf("%d ", p[i]);
	}

	free(p);
	p = NULL;

	return 0;
}

当我们输出时,结果是怎么样的呢?
在这里插入图片描述
很明显,calloc将分配给p的空间全部都进行了0的初始化

4. realloc

• realloc函数的出现让动态内存管理更加灵活。
• 有时会我们发现过去申请的空间太⼩了,有时候我们⼜会觉得申请的空间过⼤了,那为了合理的使用内存,我们⼀定会对内存的⼤⼩做灵活的调整。那 realloc 函数就可以做到对动态开辟内存⼤⼩的调整。
函数原型如下:

void* realloc(void* ptr,size_t size);

• ptr 是要调整的内存地址
• size 调整之后新⼤⼩
• 返回值为调整之后的内存起始位置。
• 这个函数调整原内存空间⼤⼩的基础上,还会将原来内存中的数据移动到新的空间。
• realloc在调整内存空间的是存在两种情况:
◦ 情况1:原有空间之后有⾜够⼤的空间
◦ 情况2:原有空间之后没有⾜够⼤的空间

在这里插入图片描述
情况1
当是情况1 的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发⽣变化。
情况2
当是情况2 的时候,原有空间之后没有⾜够多的空间时,扩展的⽅法是:在堆空间上另找⼀个合适⼤⼩的连续空间来使⽤。这样函数返回的是⼀个新的内存地址。

另外还有一个十分重要的地方,当 ptr 是 NULL 时,realloc 行为类似于 malloc(size),会分配一个大小为 size 字节的新内存块,并返回指向此内存块的指针。

所以,在使用的时候,我们一般会使用另一个指针来接受新的空间,举个例子:

#include<stdio.h>
#include<stdlib.h>

int main()
{
	int* p = (int*)malloc(5 * sizeof(int));

	if (p == NULL)//判断是否开拓失败
	{
		perror("malloc");
		return 1;
	}

	int i = 0;
	for (i = 0; i < 5; i++)
	{
		p[i] = i + 1;//放入值
	}

	int* ptr = (int*)realloc(p, 10 * sizeof(int));//用realloc拓展内存到40个字节

	if (ptr != NULL)//如果拓展成功
	{
		p = ptr;//把p指向的位置移动到ptr
		ptr = NULL;//将ptr指针置为空,之后不再使用
	}

	else
	{
		perror("realloc");//如果拓展失败
		return 1;
	}

	for (i = 5; i < 10; i++)
	{
		p[i] = i + 1;//使用拓展的内存放入数据
	}

	for (i = 0; i < 10; i++)
	{
		printf("%d ", *(p+i));//打印内存中放入的数据
	}

	free(p);//释放内存
	p = NULL;//p置为空指针

	return 0;
}

从上面这段代码可以看出,realloc拓展内存的功能十分好用,其还有缩小内存的能力,就交给读者们自己去尝试了!

如果大家看了还有什么别的想法,希望多多与我交流!

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值