C语言:动态内存管理(二)

目录

前言

1.3 realloc​编辑

3、常见动态内存管理错误

3.1 对空指针的解引用操作

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

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

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

3.5 对同一块空间的多次释放

3.6 动态内存开辟之后忘记释放

总结


前言

        接上篇继续往下将。

1.3 realloc

        realloc是调整内存空间的函数,就是用来调整内存空间的大小。参数有两个,第一个参数是指向要调整空间的起始位置;第二个参数是新的大小,希望把原来的空间调整到多大,这里就填多少(直接)。

        下面看一个例子:

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
int main()
{int* p = (int*)malloc(40);if (NULL == p){printf("%s", strerror(errno));return 1;}//使用int i = 0;for (i = 0;i < 10;i++){*(p + i) = i + 1;}for (i = 0;i < 10;i++){printf("%d ", *(p + i));}//扩容int* pn = (int*)realloc(p, 80);if (pn != NULL){p = pn;}//使用printf("\n");for (i = 0;i < 20;i++){*(p + i) = i + 1;}for (i = 0;i < 20;i++){printf("%d ", *(p + i));}return 0;
}

        在这个例子里面,我们首先用malloc建立了一块大小为40字节的空间,赋值10个整型,观察输出;后又用realloc将原有空间扩大到80字节,最后对空间赋值了20个整型,观察输出。我们来看看运行结果:

        可以看到,realloc成功将原有的40字节空间扩大到了80字节,以便我们存放20个整型数据。

        在这里要注意的是,在扩容时(也就是使用realloc函数时),我们要对新产生的指针进行判断,因为有可能扩容空间太大导致开辟不成功,计算机会返回空指针。所以有了上面的两行代码:

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

        这样能保证在扩容不成功的情况下,p指针还能保留原来40字节空间及其内容。

        关于realloc的功能,除了能对内存进行扩容以外,还可以进行空间开辟,只需要在传递指针参数的时候传递一个空指针即可。如果这么做,那么它的功能就相当于malloc。

        realloc就讲到这了。

3、常见动态内存管理错误

3.1 对空指针的解引用操作

        以前一篇里面对malloc的使用代码为例:

#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>int main()
{//动态内存开辟int* p = (int*)malloc(40);if (p == NULL)	//判断是否开辟成功,开辟失败就报错。{printf("%s\n", strerror(errno));return 1;}//使用int i = 0;for (i = 0;i < 10;i++){*(p + i) = i;}for (i = 0;i < 10;i++){printf("%d ", *(p + i));}//空间释放free(p);p = NULL;return 0;
}

        我们在使用之前,先要对定义的p进行判断,看malloc是否成功开辟空间。如果申请空间太大了超过了计算机能承担的范围,就会开辟失败,那malloc返回的将会是一个空指针,p也就为空,那我们如果不加判断直接使用p的时候,就会对其进行接引用,这样做就很危险。

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

        动态开辟的空间也是有大小的,假设我们用malloc开辟了一个40字节的空间,那么对于整型变量,最多存放10个,当我们去访问第11个整型的时候,就会造成越界访问。

        动态内存虽然叫动态内存,但它是需要我们自己去操作才能实现动态的效果的,比如上面说的,如果要去存放第11个整型,我们只需要用到calloc对原来的内存进行加长,就可以了。

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

        对于free这个函数,它只能配合函数malloc、calloc、raelloc使用,对于一般的指针指向的内容,是不能用free释放的。原因在于,malloc、calloc、realloc函数创建的内存其实和一般的变量创建的内存的位置不一样,前者在堆区,后者在栈区,他们是有区别的。一旦用free去释放一个非动态开辟的内存时,就会报错。

        例如:

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

        这样的程序一旦运行,就会报错。

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

        直接上例子:

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
int man()
{int* p = (int*)malloc(40);if (NULL == p){printf("%s", strerror(errno));return 1;}//使用int i = 0;for (i = 0;i < 10;i++){*p = i;p++;}//释放free(p);p = NULL;return 0;
}

        像这样的程序,当我们使用完p之后,p将会指向malloc开辟的空间的第十个元素的位置,这时候对p进行free是不行的,如果要用free,就要一次性将整个malloc开辟的空间进行释放,指针必须是指向这块空间的首元素的。

3.5 对同一块空间的多次释放

        这个比较好理解,我们可能会对同一块空间多次释放,这时候也是会报错的。因为第一次释放后指针指向的空间已经被回收了,但是指针任然指向原来的地址,它变成了一个野指针,再一次释放就找不到空间去释放。但如果我们在释放空间后顺便把指针指向空指针,那么再释放一次就也没有问题了。

3.6 动态内存开辟之后忘记释放

        每一次用malloc、calloc、realloc开辟动态内存的时候,在使用完成后,一定记得对其进行释放,做事情要有头有尾。如果忘记释放的话,就会造成内存泄漏,在整个程序结束前,这块内存会一直被占用,在程序结束前都无法再利用这块空间。这样的内存多了,就会导致计算机性能下降 。

总结

        关于动态内存管理的内容就讲到这。希望对你有所帮助。

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

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

相关文章

杨氏矩阵(C语言)

文章目录 问题技术名词解释思路关键代码运行代码 问题 有一个数字矩阵&#xff0c;矩阵的每行从左到右是递增的&#xff0c;矩阵从上到下是递增的&#xff0c;请编写程序在这样的矩阵中查找某个数字是否存在。 要求&#xff1a;时间复杂度小于O(N); 技术名词解释 杨氏矩阵&…

自动化测试框架Robot Framework入门

什么是RF RF是一个基于 Python 的、可扩展的关键字驱动的自动化 验收测试框架、验收测试驱动开发 &#xff08;ATDD&#xff09;、 行为驱动开发 &#xff08;BDD&#xff09; 和机器人流程自动化 &#xff08;RPA&#xff09;。它 可用于分布式、异构环境&#xff0c;其中自动…

Spring Security——04,认证_校验

认证_校验 一、认证过滤器二、配置过滤器三、测试过滤器3.1 登录接口3.2 非登录接口3.3 结果 一键三连有没有捏~~ 一、认证过滤器 创建一个filter包&#xff0c;创建一个过滤器 代码如下&#xff1a; Component public class JwtAuthenticationTokenFilter extends OncePerR…

在集群中使用deepspeed如果端口被占用可以使用deepspeed参数更改

在集群中使用deepspeed如果端口被占用可以使用deepspeed参数更改 这一次G老师不好使了 在集群中使用deepspeed默认的端口号29500被占用&#xff0c;显示更改居然不起作用 G老师给的方法也不好使 #!/bin/bash MASTER_ADDRlocalhost MASTER_PORT29501 # 选择一个未被占用的端…

mysql 运算符 语句 字符集 校队集

mysql 运算符 使用select语句可以输出运算的结果 mysql标识符不区分大小写 算数运算符 1./除法 得到的结果是一个小数 %是整数,省略小数 2、除以0不会报错,得到的结果是 null 3.数宇和字符串做加法运算,并不会拼接 比较运算符 1.mysql里面的=是比较运算符,而不是赋值运算…

lora pingpang系统 4

1 深入了解LoRa技术原理 1.1 LoRa扩频通信原理 1.1.1 模拟无线通信&#xff1a; 模拟无线通信是一种使用模拟信号传输数据的通信方式。这种通信方式已经被数字无线通信所取代&#xff0c;因为数字通信具有更高的效率和可靠性。 天线&#xff1a;从空中接收到的无线电波转换成…

数据结构——图的概念,图的存储结构,图的遍历(dfs,bfs)

目录 1.图的定义和术语 2.案例引入 1.六度空间理论 3.图的类型定义 4.图的存储结构 1.邻接矩阵 1.无向图的邻接矩阵表示法 2.有向图的邻接矩阵表示法 3.网&#xff08;有权图&#xff09;的邻接矩阵表示法 代码示例&#xff1a; 2.采用邻接矩阵表示法创建无向图…

MySQL 表的增删改查

文章目录 一、什么是CRUD&#xff1f;二、新增&#xff08;Create&#xff09;1、单行数据 全列插入2、多行数据 指定列插入3、插入特殊类型 三、查询&#xff08;Retrieve&#xff09;1、全列查询2、指定列查询3、表达式查询4、指定别名5、去重6、排序7、条件查询基本查询&a…

java算法day44 | ● 完全背包 ● 518. 零钱兑换 II ● 377. 组合总和 Ⅳ

完全背包理论基础 完全背包和01背包问题唯一不同的地方就是&#xff0c;每种物品有无限件。 体现在代码中就是对背包的遍历顺序不同。01背包是逆序遍历背包&#xff0c;完全背包是顺序遍历背包。 518. 零钱兑换 II class Solution {public int change(int amount, int[] coin…

IMAP的要点:旅行前需要了解的内容

你是否经常因会议而出差&#xff0c;需要在各种设备上灵活地访问你的电子邮件&#xff1f;如果是的话&#xff0c;你可能会想了解你的电子邮件系统是如何通过使用互联网消息访问协议&#xff08;IMAP&#xff09;来工作的&#xff0c;这样当你不在办公桌前时&#xff0c;你可以…

Java基础 - 代码练习

第一题&#xff1a;集合的运用&#xff08;幸存者&#xff09; public class demo1 {public static void main(String[] args) {ArrayList<Integer> array new ArrayList<>(); //一百个囚犯存放在array集合中Random r new Random();for (int i 0; i < 100; …

SpringCloud - 如何本地调试不会注册到线上环境(Nacos)?

问题描述 有时候我们需要本地调试注册到 Nacos 上&#xff0c;但是会影响线上服务的 Feign 请求打到本地导致不通影响了线上业务。 原因分析 一般最传统的解决方案就是修改本地 bootstrap.yml 的 spring.cloud.nacos.discovery.namespace spring:application:name: app-serv…

jvisualvm 使用教程

之前看过 jvisualvm&#xff0c;但是那个时候对 JVM 并不是很熟悉&#xff0c;后面看了下八股文&#xff0c;看了下 JVM 的相关知识之后&#xff0c;发现多了解点 JVM 的东西&#xff0c;对我们 CRUD 其实是有指导意义的&#xff0c;就比如我们通常会 new 一堆的没有用到的对象…

使用 Spring Email 和 Thymeleaf 技术,向新注册用户发送激活邮件(二)

本篇博客对应“2.2 开发注册功能”小结 对应视频: 开发注册功能 开发注册功能-续 注册功能是相对比较复制的功能&#xff0c;对于一个相对复杂的功能&#xff0c;可以把这个功能进行拆解。把这个功能的流程想清楚&#xff0c;就知道怎么拆解了&#xff1a; 也可以按照请求进行…

[每周一更]-第92期:Go项目中的限流算法

这周五在清明假期内&#xff0c;提前更新文章 很多业务会有限流的场景&#xff0c;比如活动秒杀、社区搜索查询、社区留言功能&#xff1b;保护自身系统和下游系统不被巨型流量冲垮等。 在计算机网络中&#xff0c;限流就是控制网络接口发送或接收请求的速率&#xff0c;它可防…

【算法练习】28:选择排序学习笔记

一、选择排序的算法思想 弄懂选择排序算法&#xff0c;先得知道两个概念&#xff1a;未排序序列&#xff0c;已排序序列。 原理&#xff1a;以升序为例&#xff0c;选择排序算法的思想是&#xff0c;先将整个序列当做未排序的序列&#xff0c;以序列的第一个元素开始。然后从左…

K8S - Deployment 的版本回滚

当前状态 先看deployment rootk8s-master:~# kubectl get deploy -o wide --show-labels NAME READY UP-TO-DATE AVAILABLE AGE CONTAINERS IMAGES …

理解pytorch的广播语义

目录 什么是广播运算 广播的条件 示例 示例1 示例2 示例3 补1 示例4 原位运算 示例5 参与广播运算的两个tensor&#xff0c;必须是从右向左对齐 总结规律 两个tensor可以做广播运算的条件&#xff1a; 两个可以互相广播的tensor运算的步骤&#xff1a; 例子&#x…

【深度学习】深度学习md笔记总结第3篇:TensorFlow介绍,学习目标【附代码文档】

深度学习笔记完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;深度学习课程&#xff0c;深度学习介绍要求,目标,学习目标,1.1.1 区别,学习目标,学习目标。TensorFlow介绍&#xff0c;2.4 张量学习目标,2.4.1 张量(Tensor),2.4.2 创建张量的指令,2.4.3 张量…

整数与浮点数在内存中的存储

整数与浮点数在内存中的存储 一&#xff0c;大小端存储二&#xff0c;整数在内存中的存储三&#xff0c;浮点数在内存中的存储3.1浮点数的存储规则3.2浮点数的存储过程3.2.1有效数字M3.2.2指数E3.2.3浮点数存储的特殊情况4&#xff0c;例题讲解 在C语言的编程中&#xff0c;我们…