动态内存函数详解-【malloc,calloc,realloc,free】

news2024/11/26 18:48:14

动态内存函数详解

  • malloc
    • 一、malloc的简介
      • 1.malloc函数的定义:
      • 2.参数:
      • 3.返回值:
      • 4.功能:
    • 二、malloc函数的使用
  • free函数
    • free函数的简介
    • free函数使用注意事项
  • calloc
    • calloc函数的简介
    • 二、calloc函数的使用
    • calloc函数的注意事项
  • realloc函数
    • 一、realloc函数的简介
    • 二、realloc使用
    • 三、realloc函数的注意事项
    • 四、关系

malloc

头文件:stdlib.h

一、malloc的简介

1.malloc函数的定义:

void* malloc (size_t size);

2.参数:

size: 内存块的大小(以字节为单位)。size是无符号整型。size_t

3.返回值:

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

4.功能:

向内存申请一块连续可用的空间,并返回指向这块空间的指针。单位:字节

二、malloc函数的使用

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
int main()
{
	int* p = (int*)malloc(40);//申请了40个字节,强转成int*型
	if (p == NULL)//判断是否开辟成功
	{
		perror(malloc);
		return 1;
	}
	//打印开辟空间,检验是否开辟完成
	int i = 0;
	for (i = 0;i < 10; i++)
	{
		printf("%p\n", (p + i));
	}
	free(p);
	p=NULL;
	return 0;
}

运行截图:
在这里插入图片描述
可以看到,我们已经开辟了十个整形的空间。

free函数

头文件:stdlib.h

free函数的简介

void free (void* ptr);

参数:指向先前分配的内存块的指针
如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
如果参数 ptr 是NULL指针,则函数什么事都不做。
没有返回值

free函数使用注意事项

如果开辟的空间使用完了,则应该使用free函数将p所指向的空间释放掉。
注意,p是指针变量,有自己存储空间和内容,我们free释放的只是p指向的那块空间,p并没释放,他仍然指向那块空间,所以需要手动给p=NULL。
在这里插入图片描述

calloc

头文件:stdlib.h

calloc函数的简介

void* calloc (size_t num, size_t size);

参数
1.num:要分配的元素数
2.size:要分配的元素类型
返回值
1.成功时,指向函数分配的内存块的指针。
(此指针的类型始终为 ,可以强制转换为所需类型的数据指针,以便可取消引用。)
2…如果函数无法分配请求的内存块,则返回空指针。void*
功能
–>.函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0
区别:与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

二、calloc函数的使用

#include<stdio.h>
#include<stdlib.h>
int main()
{
	int* p = (int*)calloc(10,sizeof(int));//申请了40个字节,强转成int*型
	if (p == NULL)//判断是否开辟成功
	{
		perror(malloc);
		return 1;
	}
	//打印开辟空间,检验是否开辟完成
	int i = 0;
	for (i = 0;i < 10; i++)
	{
		printf("%p\n", (p + i));
	}
	free(p);
	p = NULL;
}

在这里插入图片描述

calloc函数的注意事项

1.不能申请开辟空间太大

#include<stdio.h>
#include<stdlib.h>
int main()
{
	int* p = (int*)calloc(INT_MAX,sizeof(int));//申请了40个字节,强转成int*型
	if (p == NULL)//判断是否开辟成功
	{
		perror(calloc);
		return 1;
	}
	//打印开辟空间,检验是否开辟完成
	int i = 0;
	for (i = 0;i < 10; i++)
	{
		printf("%p\n", (p + i));
	}
	free(p);
	p = NULL;
}

在这里插入图片描述

显示错误:Not enough space

realloc函数

一、realloc函数的简介

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

1.参数
ptr:指向先前分配有 、 或 的内存块的指针。
或者,这可以是一个空指针,在这种情况下,将分配一个新块(就像被调用一样)
size:内存块的新大小(以字节为单位)。是无符号整型。size_t
(size=原来+新增加)
2.返回值:指向重新分配的内存块的指针,该内存块可能与新位置相同也可能是新位置
3.功能:(1)重新分配内存块,更改指向的内存块的大小。
(2).如果函数无法分配请求的内存块,则返回空指针,并且不会释放参数指向的内存块
(3).有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合适的内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小的调整。

二、realloc使用

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
int main()
{
	int* p = (int*)calloc(10,sizeof(int));//申请了40个字节,强转成int*型
	if (p == NULL)//判断是否开辟成功
	{
		perror(calloc);
		return 1;
	}
	//打印开辟空间,检验是否开辟完成
	int i = 0;
	for (i = 0;i < 10; i++)
	{
		p[i]=i+1;
	}
	int* ptr = (int*)realloc(p, 80);
	if (ptr != NULL)
	{
		p = ptr;
	}
	else
	{
		perror(realloc);
		return 1;//结束
	}
	for (i = 0; i < 20; i++)
	{
		printf("%d\n", p[i]);
	}
	free(p);
	p = NULL;
}

三、realloc函数的注意事项

用上面代码为例
1.使用realloc给p扩容后,不能直接用p来接收。因为一旦扩容失败,将会返回NULL,那么p所指向的内容就会直接找不到了,造成内存泄漏。应该使用一个新指针ptr来接收,判断是否扩容成功后,再将ptr指向p即可。如下:
2.free(p);就相当也释放了ptr指针。因为ptr指向了p指针。
(1)扩容成功情况

int* ptr = (int*)realloc(p, 80);
	if (ptr != NULL)
	{
		p = ptr;
	}

在这里插入图片描述
可以看到ptr和p的值相同,说明它们指向相同。即扩容成功。
(2)扩容失败情况
在这里插入图片描述
我们这扩容到8000000000000直接则失败。
在这里插入图片描述
可以见到ptr指针指向空,也说明了realloc函数开辟失败返回值是NULL。

四、关系

void* realloc (void* ptr, size_t size);
1.realloc函数的参数,如果ptr传空指针,就相当于malloc函数。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Ubuntu环境搭建

本文以Ubuntu 18.04为例 安装repo mkdir ~/bin export PATH~/bin:$PATH如果可以访问 google 的地址&#xff0c;下载 Repo 工具&#xff0c;并确保它可执行&#xff1a; curl https://storage.googleapis.com/git-repo-downloads/repo > ~/bin/repo chmod ax ~/bin/rep…

memset的坑

前言 memset 作为对内存初始化的函数&#xff0c;还是有不少坑和误区的&#xff0c;今天就来对这个函数作一个总结。 一、函数作用 memset 函数在 C 中被广泛应用于内存的初始化和设置。它可以将一段连续的内存空间快速设置为指定的值。这个函数主要作用于数组、结构体等数据类…

python 安装、配置、使用 xlrd模块

xlrd模块的分为python安装 和pycharm配置两个步骤 1. 安装xlrd模块 打开cmd&#xff0c;输入 pip install xlrd 按 enter键 安装完成即可&#xff0c;如果想升级&#xff0c;根据提示完成即可 2. pycharm 配置xlrd pycharm模块导入xlrd模块时&#xff0c;import xlrd #导入模…

【前端知识】JavaScript——var 与 let 的区别

【前端知识】JavaScript——var 与 let 的区别 var声明的变量会自动提升到函数作用域顶部&#xff0c;而let不会。 在解析代码时&#xff0c;JavaScript 引擎会注意出现在块后面的 let 声明&#xff0c;只不过在此之前不能以任何方式来引用未声明的变量。在 let 声明之前的执行…

基于simulink的DPLL仿真笔记

该笔记主要用于本人思路整理与记录 本设计运用的是电荷泵一阶环路滤波器&#xff0c;二阶三阶则在此基础上举一反三&#xff0c;以后如有机会会慢慢补全 文章目录 一.仿真模型PS&#xff08;题外话&#xff09; 二.仿真结果三.环路滤波器分析1. 环路滤波器对比LPF2. 环路滤波器…

(阿里云)STM32L+BC20+MQTT协议传输温湿度,ADC,电压,GPS数据到阿里云物联网平台

1、材料准备 准备以下材料 2、设备连接 2.1 插入物联网卡 首先把BC20核心板从开发板上拆下来 然后将物联卡放置在BC20核心板内 物联卡放置完成将BC20核心板重新插入到开发板内&#xff08;注意不要弄错方向&#xff09; 同时接入天线 2.2 连接ST-Link仿真器 用3条杜邦线接…

mzjh 项目鉴权

获取后端的接口后 将后端的数据转成数组 并报错保存 业务逻辑 function getPoints(menus) {let Points []for (let menu of menus) {if (menu.useType 2) {if (menu.grantName ! undefined && menu.grantName ! null) {Points.push(menu.grantName)}}if (menu.child…

TCP的窗口控制和重发控制【TCP原理(笔记三)】

文章目录 利用窗口控制提高速度窗口控制与重发控制确认应答未能返回的情况某个报文段丢失的情况 控制流 利用窗口控制提高速度 TCP以1个段为单位&#xff0c;每发一个段进行一次确认应答的处理&#xff0c;如图。这样的传输方式有一个缺点。那就是&#xff0c;包的往返时间越长…

YOLO-V5分类实战系列 —— 调优自己的数据集+RK1808部署

YOLO-V5分类实战系列 —— 调优自己的数据集 1、保存训练和测试图片2、数据归一化3、数据增强3.1、数据增强库&#xff1a;albumentations3.2、数据增强库&#xff1a;torchvision 4、ONNX CPU 推理4.1、Pt 模型转为 ONNX4.2、ONNX 推理验证4.3、 ONNX CPU推理&#xff08;C&am…

理解Deformable Convolution网络

1.简介 偶然了解到了可形变卷积这篇文章&#xff0c;看了几篇博文后大致了解的差不多了&#xff0c;但是有些细节还是看了代码之后才理解。这里想自己写一下关于这篇论文的了解&#xff0c;希望自己能够讲清楚。这里放一篇写的很好地博文链接&#xff0c;想更深入了解代码的可…

Gateway自定义过滤器——全局过滤器

一、什么是全局过滤器&#x1f349; 首先&#xff0c;我们要知道全局过滤器其实是特殊路由过滤器(特殊的GatewayFilter)&#xff0c;会有条件地作用于所有路由。 为什么要自定义全局过滤器&#xff1f;就好比是看大门的保安大叔&#xff0c;平时主要是做好进出大门外来人员登记…

Chatglm实现agent控制

背景&#xff1a; 这个系列文章&#xff0c;会从LLM搭建应用生态角度来写。从0到1训练一个大的通用的模型对于大部分人和团队来讲是不现实的。重资金&#xff0c;重技术含量、重投入这几个门槛可以把很多团队直接劝退。那么在LLM蓬勃发展的时候我们可以做些什么呢&#xff0c;…

C语言程序设计——字符、字符串、内存函数

一、长度不受限的字符串函数 1. strlen size_t strlen(const char* str); 功能&#xff1a;求字符串长度 &#xff08;1&#xff09;字符串以\0作为结束标志&#xff0c;strlen函数返回的是在字符串中\0之前出现的字符个数&#xff08;不包含\0&#xff09;。 &#xff08…

【每日运维】大文件的分割与合并

产生背景 特殊单位需要将文件刻盘带入&#xff0c;并且刻盘有大小限制&#xff0c;所以有了这个需求 推荐方法 个人电脑上使用 split 命令指定大小分割Linux 服务上使用 cat 命令进行合并使用 md5sum 命令校验包的完整性 方法演示 需要将一个按照100M分割后刻盘导入 在个…

数据结构 ~ 树

什么是树 - tree 一种分层数据的抽象模型&#xff1b; 如&#xff1a;DOM、级联选择、树形控件&#xff0c;js 中没有树 可以用 Object 构建树&#xff1a; const tree {val: a,children: [{val: a-1,children: [{val: a-1-1,children: []}]},{val: a-2,children: [{val: a…

mapbox绘制多边形

1、实现效果 请忽略马赛克 2、实现思路 绘制一个填充的多边形&#xff0c;再描个边框。 3、实现代码 绘制多边形函数 drawPolygon() {map.addSource(maine, {type: geojson,data: https://asc-test1.oss-cn-beijing.aliyuncs.com/2023/07/05/45f6bd80f2f34d79b3e457b31ec5d…

云原生网关如何实现安全防护能力

作者&#xff1a;刘晓瑞(钰诚) 云原生网关&#xff1a;将安全、流量和微服务三合一 作为面向南北向的公网网关&#xff0c;使用 Waf 防护异常流量是很常规的需求&#xff0c;而且随着互联网环境变得越来越复杂&#xff0c;用户对防护的诉求是持续增强的&#xff0c;常规做法是…

需要我怎么帮你?关于维护Nutsdb开源社区的思考

背景 近来有人问我打算怎么继续维护Nutsdb社区&#xff0c;他们当中不乏有开源项目的狂热爱好者&#xff0c;发起了好几个几千star的开源项目&#xff0c;也有对Nutsdb感兴趣的&#xff0c;想来问问后续的计划。这不禁让我回想到从刚开始参与这个项目到最近这段时间一个人维护…

三维重建的工作

文章目录 一、北京&#xff1a;二、广州&#xff1a;三、深圳&#xff1a; 一、北京&#xff1a; 链接 三维重建技术是自动驾驶领域4D真值数据生成的核心基础能力。融合LiDAR、Camera、IMU、轮速计等传感器数据像BlockNeRF一样重建城市级别逼真精细的三维场景&#xff0c;将…

ArcGIS Pro 矢量数据的空间校正

GIS 数据通常来自多个源。当数据源之间出现不一致时,有时需要执行额外的工作以将新数据集与其余数据进行整合。相对于基础数据而言,一些数据会在几何上发生变形或旋转。 在编辑环境中,空间校正提供用于对齐和整合数据的交互式方法。 空间校正可执行的一些任务包括:将数据…