【C语言】动态内存函数(malloc, calloc, realloc, free)

news/2024/7/20 14:12:12 标签: 指针, 内存管理, malloc, c语言

文章目录

    • 动态内存函数
      • 1)`malloc` 和 `free` 函数
      • 2)`calloc` 函数
      • 3)`realloc` 函数

本篇前言

我们之前学过的开辟空间的方式,开辟出来空间大小是固定的,往往在实际使用中,我们需要的空间大小在程序运行的时候才能知道,这个时候就需要使用到动态内存开辟了。

动态内存开辟是在内存的堆区开辟的。

image-20210725233925031


动态内存函数

malloc__free__15">1)mallocfree 函数

1、malloc 函数:

向内存堆区申请一块连续可用的空间,并返回指向这块空间的指针

void* malloc (size_t size);

size 是要开辟的内存块的大小,以字节为单位

  • 开辟成功,返回指向开辟成功的内存空间的指针类型为 void*malloc 函数并不知道开辟空间的类型,在使用的时由使用者自己来决定。

  • 开辟失败,返回空指针 NULL

2、free 函数:

释放动态开辟的内存(堆区)空间。

void free (void* ptr);

注意:free 不能用来释放栈区开辟的内存,栈区由编译器自动分配和释放,下面使用是错误的

int arr[10] = {0};
free(arr);  //错误,不能这么使用
  • malloc 和 free 的声明都在 stdlib.h 头文件中。

使用方法实例:

#include<stdio.h>
#include<stdlib.h>  /* realloc, free, NULL */
int main()
{
	//开辟10个整型空间
	int arr[10] = { 0 };  //栈区
	//动态内存开辟
	int* p = (int*)malloc(10 * sizeof(int));  //堆区
    if (p == NULL)  //对malloc函数返回值做检测
	{
		perror("malloc");  //malloc:xxxxxxxxx
		return 0;
	}
	//使用开辟的空间
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*(p + i) = i;
		//等价于:p[i] = i;
	}
	//释放开辟的空间
	free(p);  //释放开辟的空间,将其还给操作系统,但指针 p 仍然是指向开辟的空间的
	p = NULL; //手动把 p 置成NULL

	return 0;
}

注:1、在使用 malloc 函数时,一定要对其返回值做检查。2、使用完开辟的空间后,一定要记得释放空间。

如果开辟失败:perror 函数打印错误信息

image-20210725204349396

2)calloc 函数

为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为 0

void* calloc (size_t num, size_t size);
  • 开辟成功,返回指向开辟成功的内存空间的指针类型为 void*,我们可将其转换为所需类型的指针,以便解引用
  • 开辟失败,返回空指针 NULL

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

实例说明:

#include<stdio.h>
#include<stdlib.h>  /* calloc, free, NULL */
int main()
{
	//动态开辟空间
	int* p = (int*)calloc(10, sizeof(int));
	if (p == NULL)
	{
		return 0;
	}
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", *(p + i));
	}
	//释放开辟的空间
	free(p);
	p = NULL;
	return 0;
}

运行结果:

image-20210725212032001

3)realloc 函数

realloc 函数的出现让动态管理更加灵活,有时我们会发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的使用内存,我们会对内存的大小做灵活的调整。而 realloc 函数就可以做到对动态开辟内存大小的调整(扩展 / 缩小)。

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

ptr 是指向以前用 malloc、calloc 或 realloc 开辟的内存块的指针,即需要调整空间的地址

size 是调整之后的新大小,以字节为单位

  • 调整成功,返回指向调整之后的内存块的指针,可能是原始空间,也可能是新位置,类型为 void*,我们可将其转换为所需类型的指针,以便解引用
  • 调整失败,返回空指针 NULL

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

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

    image-20210725222749419
  • 情况2:原有空间之后没有足够大的连续空间

    image-20210725230314269

我们在使用 realloc 函数时一定要考虑到这两种情况

实例说明:

#include<stdio.h>
#include<stdlib.h>  /* malloc, realloc, free, NULL */
int main()
{
	//动态开辟空间
	int* p = (int*)malloc(10*sizeof(int));
	if (p == NULL)
	{
		perror("malloc");
		return 0;
	}
	//初始化
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*(p + i) = 5;
	}
	//realloc调整空间
    //p = (int*)realloc(p, 20 * sizeof(int));//可以这样写吗?
	int* ptr = (int*)realloc(p, 20*sizeof(int));
	if (ptr != NULL)  //检查空间是否调整成功
	{
		p = ptr;
	}
	//释放空间
	free(p);
	p = NULL;
	return 0;
}

注:不要用指向原始空间的指针 p 来接收,因为 realloc 有可能找不到合适的空间来调整大小,这时会返回 NULL,我们一般用一个临时指针 ptr 来接收,然后检查空间是否调整成功,如果调整成功,再将 p = ptr

image-20210725225640787

运行结果:

情况1:原有空间之后有足够大的连续空间,返回原位置

image-20210726112403361

情况2:原有空间后没有足够大的连续空间,返回新位置

image-20210726103636845


补充:

#include<stdlib.h>  /* realloc, free, NULL */
int main()
{
	//这里功能类似于 malloc,就是直接在堆区开辟了40个字节的空间
	int* p = (int*)realloc(NULL, 40);
	if (p == NULL)
	{
		perror("realloc");
		return 0;
	}
	free(p);
	p = NULL;
	return 0;
}

注意:

再次叨叨下,

使用动态内存开辟函数 malloccallocrealloc 时,一定要记得,

将其转换为我们所需类型的指针

然后再检查是否(为NULL)开辟成功了,

最后记得 free 释放空间并置空 NULL。



http://www.niftyadmin.cn/n/1579750.html

相关文章

latex---笑死我了

老美幽默死了个人&#xff01;转载于:https://www.cnblogs.com/papershop/archive/2012/12/02/2798112.html

【数据结构入门】算法的时间复杂度和空间复杂度详解

文章目录&#xff08;1&#xff09;算法效率&#xff08;2&#xff09;时间复杂度的计算1&#xff09;什么是时间复杂度2&#xff09;大O渐进表示法&#xff08;估算&#xff09;3&#xff09;时间复杂度计算实例4&#xff09;总结5&#xff09;一些思考&#xff08;3&#xff…

【LeetCode】数组nums包含从0到n的所有整数,但其中缺了一个,请找出那个缺失的整数(面试题 17.04. 消失的数字)| 数组

文章目录1、题目描述&#xff1a;2、解题方法&#xff1a;解法1&#xff1a;解法2&#xff1a;解法3&#xff1a;1、题目描述&#xff1a; LeetCode链接&#xff1a;消失的数字 2、解题方法&#xff1a; 对于此道题&#xff0c;我们有哪些解题思路呢&#xff1f;先不管其事件…

网站架构相关PPT、文章整理(更新于2009-7-15)

在这篇blog中放置了我收集的一些网站架构相关的PPT和文章&#xff0c;提供给大家下载&#xff0c;如果大家有相关的好的PPT、文章的话&#xff0c;也欢迎推荐给我&#xff0c;非常感谢&#xff0c;:)&#xff0c;这篇blog的内容也会随着我收集的东西增加而变化&#xff0c;同时…

【LeetCode】原地将数组中的元素向右移动 k 个位置(189. 旋转数组)| 数组、三部翻转法

文章目录1、题目描述&#xff1a;2、解题方法&#xff1a;解法1&#xff1a;解法2&#xff1a;解法3&#xff1a;1、题目描述&#xff1a; LeetCode链接&#xff1a;https://leetcode-cn.com/problems/rotate-array/ 2、解题方法&#xff1a; 解法1&#xff1a; 右旋一次&am…

char 与 String 相等比较

这是一个相当2 相当基础 相当没有意义的帖子&#xff1b;但今天因为这个问题引发了一个bug.小细节也很重要&#xff01;&#xff01;&#xff01; char a1;// char b2dsf; //char 表示单个字符 char c1; //不需要单引号也能表示// System.out.println&#xff08;a.equals&…

【LeetCode】原地移除数组中数值等于 val 的所有元素(27. 移除元素)| 数组、双指针(快慢指针) | 动图演示

文章目录1、题目描述&#xff1a;2、解题方法&#xff1a;1、题目描述&#xff1a; LeetCode链接&#xff1a;移除元素 2、解题方法&#xff1a; 双指针&#xff08;快慢指针&#xff09;&#xff0c;将数组中要移除的元素用后面不等于 val 的元素依次覆盖掉。 相当于把不等于…

使用SQL语句操作数据库

(一)ADO.NET与数据库交互的两种方式•面向连接方式•非面向连接方式•面向连接方式操作数据库的步骤&#xff1a;1.建立Connetion对象以连接数据库2.建立Command对象3.设置Command对象的CommandText4.打开Connetion的连接5.运行SQL语句6.生成DataReader对象&#xff0c;并利用该…