C语言——动态内存管理 malloc、calloc、realloc、free的使用

目录

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

二、动态内存函数的介绍

2.1malloc和free

2.2calloc

2.3realloc

三、常见的动态内存错误

3.1对NULL指针的解引用操作

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

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

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

3.5对同一块内存多次释放

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

四、几个经典的题


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

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

int a = 10;//在栈空间开辟四个字节

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

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

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

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

但是对于空间的需求,不仅仅是上述情况,有时候我们需要的空间大小在程序运行的时候才能知道,如此数组的编译时开辟空间的方式就不能满足了。这个时候可以使用动态开辟了。

二、动态内存函数的介绍

2.1malloc和free

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

void* malloc(size_t size);

  • 如果开辟成功,则返回一个指向开辟好的空间的指针
  • 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查
  • 返回值的类型是void*,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
  • 如果参数size为0,malloc的行为是标准是未定义的,取决于编译器

//C语言提供了另外一个函数free,专门就用来做动态内存的释放和回收的

//free函数用来释放动态开辟的内存

void free(void* ptr);

如果参数ptr指向的空间不是动态开辟的,那free函数的行为是未定义的

如果参数ptr是NULL指针,则函数声明事都不做。

int main()
{
	//申请
	int* str = (int*)malloc(sizeof(int) * 5);
	if (str == NULL)
	{
		printf("malloc %s", strerror(errno));
		exit(-1);
	}
	//使用
	for (int i = 0; i < 5; i++)
	{
		*(str +i) = i + 1;
	}
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", *(str +i));
	}
	//释放
	free(str);
	str = NULL;

	return 0;
}

2.2calloc

void* calloc(size_t num, size_t size);

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

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

int main()
{
	//创建了一块大小为10个大小为int类型的空间,默认初始化成0
	int* p = (int*)calloc(10, sizeof(int));
	if (p == NULL)
	{
		printf("calloc:%s", strerror(errno));
		exit(-1);
	}
	
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", *(p+i));
	}
	//释放
	free(p);
	p = NULL;
	return 0;
}

2.3realloc

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

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

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

  • ptr是要调整的内存地址;
  • size调整之后新大小;
  • 返回值为调整之后的内存起始位置;
  • 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间;
  • realloc在调整内存空间的是存在两种情况:

1.原有空间之后有足够大的空间:要扩展内存就直接在原有内存之后直接追加空间,原来空间的数据不会发生变化。

2.原有空间之后没有足够的空间:在堆空间上另找一个合适大小的连续空间来使用。这样函数返回的是一个新的内存地址。

int main()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	if (p == NULL)
	{
		printf("malloc:%s\n", strerror(errno));
		exit(-1);
	}
	for (int i = 0; i < 5; i++)
	{
		*(p + i) = i + 1;
	}
	for (int i = 0; i < 5; i++)
	{
		printf("%d ", *(p+i));
	}
	int* ptr = (int*)realloc(p, sizeof(int) * 10);
	if (ptr == NULL)
	{
		printf("realloc:%s\n", strerror(errno));
		exit(-1);
	}
	else
	{
		p = ptr;
		for (int i = 5; i < 10; i++)
		{
			p[i] = i + 1;
		}
		for (int i = 0; i < 10; i++)
		{
			printf("%d ", p[i]);
		}
	}
	//释放
	free(p);
	p = NULL;
	return 0;
}

三、常见的动态内存错误

3.1对NULL指针的解引用操作

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

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

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

 

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

void test3()
{
	int a = 10;
	int* p = &a;
	//对非动态开辟的内存使用free释放
	free(p);
}

 

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

void test4()
{
	int* p = (int*)malloc(100);
	p++;//p指向后面的地址
	free(p);//必须提供起始地址的地址来free
}

3.5对同一块内存多次释放

void test5()
{
	int* p = (int*)malloc(100);
	free(p);
	free(p);//重复释放
}

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

malloc、calloc、realloc等所申请的空间不想使用需要free释放,如果不使用free释放程序结束之后,也会由操作系统回收,如果不使用free释放,程序也不结束,那么会造成内存泄漏

void test6()
{
	int* p = (int*)malloc(100);
	if (p != NULL)
	{
		*p = 20;
	}
}


int main()
{
	test6();//动态开辟的内容忘记释放(内存泄漏)
	while (1);
	return 0;
}

四、几个经典的题

//传值调用,不会影响str,str依然为NULL,
//1.strcpy函数调用失败,原因是对NULL的解引用操作,程序会崩溃
//2.没有释放,会造成内存泄漏
//
void GetMemory1(char* p)
{
	p = (char*)malloc(100);
}
void test7()
{
	char* str = NULL;
	GetMemory1(str);
	strcpy(str, "hello world");
	printf(str);
}
//GetMemory2函数内部创建的数组是临时的,虽然返回了p给str,但数组的内存出了函数就会归还给操作系统,
//而str依然保持了数组的起始地址,这时如果使用str, str就是野指针
//
char* GetMemory2()
{
	char p[] = "hello world";
	return p;
}
void test8()
{
	char* str = NULL;
	str = GetMemory2();
	printf(str);
}
//传址调用,str指向malloc分配出来的起始地址,但是最后没有释放,会造成内存泄漏
void GetMemory3(char** p, int num)
{
	*p = (char*)malloc(num);
}

void test9()
{
	char* str = NULL;
	GetMemory3(&str, 100);
	strcpy(str, "hello");
	printf(str);
	//释放
	//free(str);
}
//在free后没有将str置空,str指向的内存空间被还给操作系统了,此时str是野指针,往str里拷贝字符串会形成非法访问
void test10()
{
	char* str = (char*)malloc(100);
	strcpy(str, "hello");
	free(str);//free完后要将str置为空
	str = NULL;
	if (str != NULL)
	{
		strcpy(str, "world");
		printf(str);
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/2877.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

奇安信_防火墙部署_透明桥模式

奇安信_防火墙部署_透明桥模式一、预备知识二、项目场景三、拓扑图四、基本部署配置1. 登录web控制台2.连通性配置3.可信主机配置4.授权导入5.特征库升级6.安全配置文件五、透明桥配置1. 创建桥2. 端口绑定桥3. 创建桥端口六、结语一、预备知识 安全设备接入网络部署方式 二、…

运算放大器:电压比较器

目录一、单限电压比较器二、滞回电压比较器三、窗口电压比较器最近在学习电机控制&#xff0c;遇到了与运算放大电路相关的知识&#xff0c;然而太久没有接触模拟电路&#xff0c;对该知识已经淡忘了&#xff0c;及时温故而知新&#xff0c;做好笔记&#xff0c;若有错误、不足…

字节跳动测试岗面试记:二面被按地上血虐,所幸Offer已到手...

在互联网做了几年之后&#xff0c;去大厂“镀镀金”是大部分人的首选。大厂不仅待遇高、福利好&#xff0c;更重要的是&#xff0c;它是对你专业能力的背书&#xff0c;大厂工作背景多少会给你的简历增加几分竞争力。 但说实话&#xff0c;想进大厂还真没那么容易。最近面试字…

3分钟阐述这些年我的 接口自动化测试 职业生涯经验分享

接口自动化测试学习教程地址&#xff1a;https://www.bilibili.com/video/BV1914y1F7Bv/ 你好&#xff0c;我是凡哥。 很高兴能够分享我的接口自动化测试经验和心得体会。在我目前的职业生涯中&#xff0c;接口自动化测试是我经常进行的一项任务。通过不断地学习和实践&#xf…

【C++】map 和 set

文章目录一、关联式容器与键值对1、关联式容器2、键值对 pair3、树形结构的关联式容器二、set1、set 的介绍2、set 的使用三、multiset四、map1、map 的介绍2、map 的使用五、multimap一、关联式容器与键值对 1、关联式容器 在C初阶的时候&#xff0c;我们已经接触了 STL 中的…

基于SpringBoot的酒店管理系统

系统环境 开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/i…

matplotlib参数详解

文章目录一、简介二、安装与调用三、绘图与风格设置3.1、绘图标记3.1.1、标记类型&#xff08;marker*&#xff09;3.1.2、标记大小、内部和边框颜色&#xff08;ms10、mfcr、mecg&#xff09;3.2、绘图线3.2.1、线类型&#xff08;linestyle--&#xff09;3.2.2、线宽&#xf…

C++入门教程||C++ 字符串||

C 字符串C 字符串C 提供了以下两种类型的字符串表示形式&#xff1a;C 风格字符串C 引入的 string 类类型C 风格字符串C 风格的字符串起源于 C 语言&#xff0c;并在 C 中继续得到支持。字符串实际上是使用 null 字符 终止的一维字符数组。因此&#xff0c;一个以 null 结尾的…

大文件上传

上图就是大致的流程一、标题图片上传课程的标题图片Ajax发送请求到后端后端接收到图片使用IO流去保存图片&#xff0c;返回图片的信息对象JS回调函数接收对象通过$("元素id").val(值)&#xff0c;方式给页面form表达img标签src属性值&#xff0c;达到上传图片并回显二…

若依微服务(ruoyi-cloud)保姆版容器编排运行

一、简介 项目gitee地址&#xff1a;https://gitee.com/y_project/RuoYi-Cloud 由于该项目运行有很多坑&#xff0c;大家可以在git克隆拷贝到本地后&#xff0c;执行下面的命令使master版本回退到本篇博客的版本&#xff1a; git reset --hard 05ca78e82fb4e074760156359d09a…

扒一扒抖音是如何做线程优化的

背景 最近在对一些大厂App进行研究学习&#xff0c;在对某音App进行研究时&#xff0c;发现其在线程方面做了一些优化工作&#xff0c;并且其解决的问题也是之前我在做线上卡顿优化时遇到的&#xff0c;因此对其具体实现方案做了深入分析。本文是对其相关源码的研究加上个人理…

人员玩手机离岗识别检测系统 yolov5

人员玩手机离岗识别检测系统根通过pythonyolov5网络模型识别算法技术&#xff0c;人员玩手机离岗识别检测算法可以对画面中人员睡岗离岗、玩手机打电话、脱岗睡岗情况进行全天候不间断进行识别检测报警提醒。Python是一种由Guido van Rossum开发的通用编程语言&#xff0c;它很…

【2023.3.18 美团校招】

文章目录1. 小美剪彩带2. 最多修改两个字符&#xff0c;生成字典序最小的回文串1. 小美剪彩带 题意&#xff1a;找出区间内不超过k种数字子数组的最大长度 使用双指针的方式&#xff0c;用哈希表来统计每个数出现次数。在双指针移动的过程中&#xff0c;动态的维护区间内不同数…

难以置信,已经有人用 ChatGPT 做 Excel 报表了?

要问2023年初科技领域什么最火&#xff0c;那自然是 ChatGPT。 ChatGPT 由人工智能研究实验室 OpenAI 于2022年11月30日推出。上线短短5天&#xff0c;用户数量已突破100万&#xff0c;在今年2月份&#xff0c;用户数量已经突破1亿。 ChatGPT 是一个超级智能聊天机器人&#…

【java】笔试强训Day4【计算糖果、进制转换】

目录 ⛳一、单选题 1.下列与队列结构有关联的是&#xff08; &#xff09; 2.类所实现接口的修饰符不能为&#xff08; &#xff09; 3.下列关于栈叙述正确的是&#xff08; &#xff09; 4.下面关于abstract关键字描述错误的是&#xff08; …

9.0 自定义SystemUI下拉状态栏和通知栏视图(十五)之悬浮通知布局

1.前言在进行9.0的系统rom产品定制化开发中&#xff0c;在9.0中针对systemui下拉状态栏和通知栏的定制UI的工作开发中&#xff0c;原生系统的下拉状态栏和通知栏的视图UI在产品开发中会不太满足功能&#xff0c;所以根据产品需要来自定义SystemUI的下拉状态栏和通知栏功能&…

【jenkins部署】一文弄懂自动打包部署(前后台)

这里写目录标题序言软件安装jdkmaven配置maven阿里镜像以及本地库位置git安装安装jenkins插件安装环境配置创建项目配置gitee生成gitee WebHookmaven打包验证是否打包成功连接远程服务器并重启服务远程服务器生成私钥配置ssh项目配置ssh脚本vue项目打包nodejs安装下载配置环境变…

【C++进阶】位图和布隆过滤器

文章目录位图位图概念位图使用场景位图的结构构造setresettest完整代码布隆过滤器布隆过滤器概念布隆过滤器结构构造setresettest完整版代码位图 位图概念 所谓位图&#xff0c;就是用每一位来存放某种状态&#xff0c;适用于海量数据&#xff0c;数据无重复的场景。通常是用…

智能火焰与烟雾检测系统(Python+YOLOv5深度学习模型+清新界面)

摘要&#xff1a;智能火焰与烟雾检测系统用于智能日常火灾检测报警&#xff0c;利用摄像头画面实时识别火焰与烟雾&#xff0c;另外支持图片、视频火焰检测并进行结果可视化。本文详细介绍基于智能火焰与烟雾检测系统&#xff0c;在介绍算法原理的同时&#xff0c;给出Python的…

【数据结构与算法】设计循环队列

文章目录&#x1f451;前言如何设计循环队列设计循环队列整体的代码&#x1f4ef;写在最后&#x1f451;前言 &#x1f6a9;前面我们 用队列实现了一个栈 &#xff0c;用栈实现了一个队列 &#xff0c;相信大家随随便便轻松拿捏&#xff0c;而本章将带大家上点难度&#xff0c;…