【C语言】动态内存管理,详细!!!

文章目录

  • 前言
  • 一、为什么存在动态内存分配
  • 二、动态内存开辟函数的介绍
    • 1.malloc
    • 2.calloc
    • 3.realloc
    • 4.free
  • 三、动态内存开辟中的常见错误
    • 1.误对NULL进行解引用操作
    • 2.对于动态开辟的空间进行了越界访问
    • 3.对于非动态开辟的内存进行了free操作
    • 4.只free掉动态开辟内存的一部分
    • 5.多次free已经释放的空间内存
  • 四、总结


添加链接描述

前言

大家好呀,时隔好几天小小樊又来为大家分享C语言学习啦,今天为大家分享一下自己对于动态内存管理的理解!!!

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

对于栈上开辟的空间:

  1. 空间开辟大小是固定的。
  2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了

这时就需要在堆区矩形动态内存开辟了

二、动态内存开辟函数的介绍

1.malloc

大家先看一下库中对于他的说明:
在这里插入图片描述

函数功能:开辟内存块
参数size_t:需要申请的字节数
返回值:申请失败返回空指针,申请成功返回指向申请该空间首地址的指针头文件:stdlib.h
注意:返回指针的类型是void*,这时候需要你把该指针强制类型转化为你想要的类型,这样方便访问,以及解引用,malloc申请来的空间是连续的,但是多次malloc来的是不连续的
malloc的使用

int main()
{
	int*p=(int*) malloc(40);//申请了40个字节,强制转化为int*类型指针
	if (p == NULL)//如果返回空指针的话,申请失败
	{
		perror("malloc:");//打印错误信息
		return 1;//非正常退出
	}
	for (int i = 0; i < 10; i++)
	{
		*(p + i) = i;//对每一个四个字节大小的元素赋值,这里*(p+i)的本质就是p[i];
		printf("%d", *(p + i));//打印每个元素
	}
	return 0;//程序正常退出


}

2.calloc

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:申请一个数组在内存中,并且初始化为0;
参数:size_t num申请数组元素的个数,size_t size每个元素的字节大小
返回值:申请失败返回空指针,申请成功返回指向申请该空间首地址的指针头文件:stdlib.h
calloc函数的使用:

int main()
{
	int i = 0;
	int*p=(int*) calloc(10,sizeof(int));//申请10个元素,每个元素字节大小4
	if (p == NULL)//如果返回空指针的话,申请失败
	{
		perror("calloc:");//打印错误信息
		return 1;//非正常退出
	}
	for (int i = 0; i < 10; i++)
	{
		
		printf("%d ", *(p + i));//打印初始化的值
	}
	free(p);
	p = NULL;
    return 0;


}

在这里插入图片描述
malloc和calloc的区别:
与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

3.realloc

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:内存块的扩容
参数:第一个参数接收要扩容内存块的首地址,扩容后总字节大小(包括原来的字节大小)
头文件:stdlib.h
初始扩容的空间为空,则realloc和malloc的用法一模一样
返回值:扩容后空间的首地址
在这里插入图片描述
在这里插入图片描述
realloc函数的使用:

int main()
{
	int* p = (int*)malloc(40);//申请了40个字节,强制转化为int*类型指针
	if (p == NULL)//如果返回空指针的话,申请失败
	{
		perror("malloc:");//打印错误信息
		return 1;//非正常退出
	}
	for (int i = 0; i < 10; i++)//循环打印扩容前的元素
	{
		*(p + i) = i;
		printf("%d ", *(p + i));
	}
	int* ptr = (int*)realloc(p, 80);//原空间够用ptr==p,不够用的话ptr存放新地址
	if (ptr != NULL)//扩容成功
	{
		p = ptr;//原空间够用ptr==p,不够用的话ptr存放新地址,重新将新地址给p
	}
	for (int i = 10; i < 20; i++)//扩容后新空间的
	{
		*(p + i) = i;
		printf("%d ", *(p + i));
	}

    free(p);
	p = NULL;
	
	return 0;
}

在这里插入图片描述

4.free

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:释放内存块
参数:指针接收要释放内存块的首地址
头文件:stdlib.h
返回值:无

注意:
当p所指向的申请的空间释放时,p指针指向随机位置,p变成野指针,所以我们要在释放后将其置为空!!!

如果我们不释放动态内存申请的内存的时候,程序结束,动态申请内存由操作系统自动回收,如果不用free函数释放申请好的空间,就会在程序运行结束前一直存在于堆中,造成内存泄漏

三、动态内存开辟中的常见错误

1.误对NULL进行解引用操作

比如:

int main()
{
	int* p = (int*)malloc(1000);
	int i = 0;
	if (p ==NULL)
	{
		return 1;
	}
	for (i = 0; i < 250; i++)
	{
		*(p + i) = i;
	}
	free(p);
	p = NULL;
	return 0;
}

当开辟内存失败时会返回空,这时容易造成此错误。
解决方法:开辟内存后进行判断,如上面代码中的 if 判断

2.对于动态开辟的空间进行了越界访问

int main()
{
	int* p = (int*)malloc(100);
	int i = 0;
	if (p ==NULL)
	{
		return 1;
	}
	for (i = 0; i <=25; i++)//越界访问
	{
		*(p + i) = i;
	}
	free(p);
	p = NULL;
	return 0;
}

解决方法:人为进行检查是否越界

3.对于非动态开辟的内存进行了free操作

int main()
{
	int a = 10;
	int* p = &a;
	free(p);
	p = NULL;
	return 0;
}

4.只free掉动态开辟内存的一部分

int main()
{
	int* p = (int*)malloc(100);
	if (p == NULL)
	{
		return 1;}
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*p = i;
		p++;
    }
	free(p);
	p = NULL;
	return 0;
}

解决方案:
别改变保存动态开辟空间首地址的指针变量,使用时可以采用中间变量的方法!!!
例如:

int main()
{
	int* p = (int*)malloc(100);
	if (p == NULL)
	{
		return 1;


	}
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*(p+i)= i;
		printf("%d ", *(p + i));
    }
	free(p);
	p = NULL;

	return 0;
}

5.多次free已经释放的空间内存

int main()
{
	int* p = malloc(40);
	if (p == NULL)
	{
		return 1;
	}
	free(p);
	free(p);
	p = NULL;
	return 0;
}

四、总结

本次内容到这里就分享完啦,如果大家觉得对自己有帮助的话还请大家点个赞呀,有分享不对的地方还恳请大家指正,谢谢大家的阅读!!!

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

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

相关文章

Linux工具【2】(调试器gdb、项目自动化构建工具make/Makefile)

gdb、make/Makefile 引言调试器gdb介绍常用指令 自动化构建工具make/Makefile介绍使用依赖关系与依赖方法编辑Makefile伪目标 总结 引言 在上一篇文章中介绍了Linux中的编辑器vim与编译器gcc与g&#xff1a; 戳我看vim与gcc详解哦 在本篇文章中将继续来介绍Linux中的工具&…

ResNet18云空间部署

1-6步骤可以在云空间运行&#xff0c;也可以在本地运行&#xff1b;步骤7 在云空间运行。 1.编译ONNX模型 本章以 resnet18.onnx 为例, 介绍如何编译迁移一个onnx模型至BM1684X TPU平台运行。 该模型来自onnx的官网: models/vision/classification/resnet/model/resnet18-v1…

Mybatis (3)-----分页的运用

目录 一、分页查询 二&#xff0c;特殊的字符处理 三、总结 前言&#xff1a;在我们上篇已经学的动态sql的基础上&#xff0c;今天继续讲解关于maybatis的分页&#xff0c;特殊的字符处理。希望这篇博客可以帮助到大家哦&#xff01; 一、分页查询 为什么要重写mybatis的分…

星际争霸之小霸王之小蜜蜂(六)--让子弹飞

目录 前言 一、添加子弹设置 二、创建子弹 三、创建绘制和移动子弹函数 四、让子弹飞 五、效果 总结 前言 小蜜蜂的基本操作已经完成了&#xff0c;现在开始编写子弹的代码了。 一、添加子弹设置 在我的预想里&#xff0c;我们的小蜜蜂既然是一只猫&#xff0c;那么放出的子弹…

【MD5加密】

MD5加密 什么是MD5密码MD5用途MD5特点MD5加密MD5解密总结那我们上面也已经提到啦&#xff0c;说MD5是可以进行解密或者说他是可以泄露密码等&#xff0c;所以我们还可以使用以下方法进行再次加密 第一种&#xff1a;MD5固定盐值第二种&#xff1a;MD5随机盐值 什么是MD5密码 官…

基于鸽群算法优化的BP神经网络(预测应用) - 附代码

基于鸽群算法优化的BP神经网络&#xff08;预测应用&#xff09; - 附代码 文章目录 基于鸽群算法优化的BP神经网络&#xff08;预测应用&#xff09; - 附代码1.数据介绍2.鸽群优化BP神经网络2.1 BP神经网络参数设置2.2 鸽群算法应用 4.测试结果&#xff1a;5.Matlab代码 摘要…

25-非父子通信 - event bus 事件总线

作用:非父子组件之间&#xff0c;进行简易消息传递。(复杂场景 -> Vuex) 1.创建一个都能访问到的事件总线(空 Vue 实例) -> utils/EventBus.js import Vue from vue const Bus new Vue() export default Bus 2. A组件(接收方),监听 Bus实例 的事件(支持多个组件同时监…

机器人制作开源方案 | 滑板助力器

我们可以用一块废滑板做些什么呢&#xff1f; 如今&#xff0c;越来越多的人选择电动滑板作为代步工具或娱乐方式&#xff0c;市场上也涌现出越来越多的电动滑板产品。 &#xff08;图片来源&#xff1a;Backfire Zealot X Belt Drive Electric Skateboard– Backfire Board…

在Linux系统上安装和配置Redis数据库,无需公网IP即可实现远程连接的详细解析

文章目录 1. Linux(centos8)安装redis数据库2. 配置redis数据库3. 内网穿透3.1 安装cpolar内网穿透3.2 创建隧道映射本地端口 4. 配置固定TCP端口地址4.1 保留一个固定tcp地址4.2 配置固定TCP地址4.3 使用固定的tcp地址连接 Redis作为一款高速缓存的key value键值对的数据库,在…

学会Mybatis框架:一文掌握MyBatis与GitHub插件分页的完美结合【三.分页】

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Mybatis的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.Mybatis分页 1. Mybatis自带分页 2…

springboot小知识:配置feign服务超时时间

背景&#xff1a;当前项目通过feign服务调用了其他两个项目的接口&#xff0c;但是由于特殊需求&#xff0c;需要调整某一个项目的feign服务的默认超时时间&#xff1a; 默认连接超时10秒&#xff0c;默认读取超时时间 60秒 1.找到定义的FeignClient 2.根据FeignClient定义的名…

Node基础--Node简介以及安装教程

1.Node简介 Node.js发布于2009年5月&#xff0c;由Ryan Dahl开发&#xff0c;是一个基于Chrome V8引擎的JavaScript运行环境&#xff0c;使用了一个事件驱动、非阻塞式I/O模型&#xff0c;让JavaScript 运行在服务端的开发平台&#xff0c;它让JavaScript成为与PHP、Python、Pe…

对CSV格式的数据文件进行插值处理

使用Python程序&#xff0c;实现对一个较短的csv文件&#xff0c;进行差值处理&#xff0c;并绘制GUI界面&#xff1b; 这个程序是一个使用Python的Tkinter库构建的GUI应用程序&#xff0c;用于对CSV格式的数据文件进行插值处理。下面我会逐步解释程序的各个部分和功能&#x…

消息队列前世今生 字节跳动 Kafka #创作活动

消息队列前世今生 1.1 案例一&#xff1a; 系统崩溃 首先大家跟着我想象一下下面的这个的场景&#xff0c; 看到新出的游戏机&#xff0c;太贵了买不起&#xff0c;这个时候你突然想到&#xff0c;今天抖音直播搞活动&#xff0c;打开抖音搜索&#xff0c;找到直播间以后&am…

边缘计算节点BEC典型实践:如何快速上手PC-Farm服务器?

百度智能云边缘计算节点BEC&#xff08;Baidu Edge Computing&#xff09;基于运营商边缘节点和网络构建&#xff0c;一站式提供靠近终端用户的弹性计算资源。边缘计算节点在海外覆盖五大洲&#xff0c;在国内覆盖全国七大区、三大运营商。BEC通过就近计算和处理&#xff0c;大…

并发编程系列-分而治之思想Forkjoin

我们介绍过一些有关并发编程的工具和概念&#xff0c;包括线程池、Future、CompletableFuture和CompletionService。如果仔细观察&#xff0c;你会发现这些工具实际上是帮助我们从任务的角度来解决并发问题的&#xff0c;而不是让我们陷入线程之间如何协作的繁琐细节&#xff0…

一网打尽java注解-克隆-面向对象设计原则-设计模式

文章目录 注解内置注解元注解 对象克隆为什么要克隆&#xff1f;如何克隆浅克隆深克隆 Java设计模式什么是设计模式&#xff1f;为什么要学习设计模式&#xff1f; 建模语言类接口类之间的关系依赖关系关联关系聚合关系组合关系继承关系实现关系 面向对象设计原则单一职责开闭原…

仓库使用综合练习

目录 1、使用mysql:5.6和 owncloud 镜像&#xff0c;构建一个个人网盘。 2、安装搭建私有仓库 Harbor 3、编写Dockerfile制作Web应用系统nginx镜像&#xff0c;生成镜像nginx:v1.1&#xff0c;并推送其到私有仓库。 4、Dockerfile快速搭建自己专属的LAMP环境&#xff0c;生…

跨平台图表:ChartDirector for .NET 7.1 Crack

什么是新的 ChartDirector for .NET 7.0 支持跨平台使用&#xff0c;但仅限于 .NET 6。这是因为在 .NET 7 中&#xff0c;Microsoft 停止了用于非 Windows 使用的 .NET 图形库 System.Drawing.Common。由于 ChartDirector for .NET 7.0 依赖于该库&#xff0c;因此它不再支持 .…

小白到运维工程师自学之路 第七十九集 (基于Jenkins自动打包并部署Tomcat环境)1

一、传统的流程 1、传统网站部署的流程 在运维过程中&#xff0c;网站部署是运维的工作之一。传统的网站部署的流程大致分为:需求分 析-->原型设计-->开发代码-->提交代码-->内网部署-->内网测试-->确认上线-->备份数据-->外网更新-->外网测试--&g…