C语言动态内存管理

我们目前知道的开辟内存空间的方法有:  1.创建变量   2.创建数组;

但是这2种方法开辟的空间大小都是固定的,如果是数组的话确认了大小之后是无法改变的;

int a=10;//在栈区空间上开辟4个字节的空间;

int arr[10];//在栈区空间上开辟10个整型的空间;

有了动态开辟可以让我们更加灵活的运用内存空间,成为我们有力的武器;

1.malloc和free

malloc这个函数可以让我们指定开辟内存空间的大小,它的参数是size_t的num;这个参数是要开辟的空间多大,单位是字节;它的返回值是开辟好的内存空间的起始地址;

但是,malloc不总是会开辟成功,也会开辟失败,当开辟空间失败它会返回一个空指针的,所以我们在使用的时候一定要去判断它的返回值,不然我们可能会对空指针进行操作;

练习:

利用malloc函数创建一个二维数组;

int main()
{
	int(*arr)[5] = (int (*)[5])malloc(15 * sizeof(int));
	if (arr == NULL)
	{
		perror("malloc");
		return 1;
	}
	int i = 0;
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			scanf("%d", &arr[i][j]);
		}
	}
	printf("\n");
	for (i = 0; i < 3; i++)
	{
		int j = 0;
		for (j = 0; j < 5; j++)
		{
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
	free(arr);
	arr = NULL;
	return 0;
}

free是和动态内存开辟函数搭配使用的,当我们开辟好一个空间,使用完之后我们不再需要这块空间的时候,要及时将空间回收给操作系统;这个函数就是用来回收动态内存开辟的空间的,它的参数是要回收的那块空间的起始地址;

如果我们给这个函数传空指针的话它什么也不会做,同时,将空间回收之后要把指向那块空间起始地址的指针及时置为空指针,不然会变成野指针;

2.calloc

calloc这个函数用法跟malloc差不多,参数是num个大小为size类型的字节的空间;它如果开辟成功则返回开辟好的空间的起始地址,开辟失败返回空指针;

但是calloc开辟的内存空间会全部初始化为0;用法上和malloc一样;

3.realloc

realloc是用来调整动态开辟内存空间的大小的;参数部分ptr是要调整的那块空间的起始地址;size是调整后的空间的大小;如果开辟成功则返回开辟好的空间的起始位置,失败则返回空指针;

malloc函数开辟空间成功的话返回值有2种情况:

第一种:后面的空间刚好足够新的空间的大小

这个时候会返回旧的起始位置的地址;

第二种:后面容纳不下新增空间的位置

此时realloc就会重新找到一个新的可以容纳的下新空间大小的内存;并把旧的空间的数据先复制到新的空间去,然后再把旧的空间还给操作系统;此时realloc的返回的就是新的空间的地址;

在使用realloc的时候一般创建一个临时的指针变量来接收它的返回值,然后进行判断,如果不为空指针的话就把临时指针变量的值重新赋值给旧的那个指针变量;

举例说明:

int main()
{
	int* pa = (int*)malloc(40);
	if (pa == NULL)
	{
		return 1;
	}
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		pa[i] = i;
	}
	int * ps = (int *)realloc(pa, 20*sizeof(int));
	if (pa == NULL)
	{
		return 1;
	}
	else
	{
		pa = ps;
	}
	//继续使用
	free(pa);
	ps = NULL;
	pa = NULL;
	return 0;
}

我们将新增的空间设置大一点:

 我们可以看到ps和pa的地址是不一样的,这也符合了第二个结论;

realloc的第一个参数如果给它传空指针的话,它的效果和malloc'的效果一样;

以上三个函数所创建的内存空间都是在内存的堆区上的,堆区上的空间只能由我们自己释放空间,或者程序结束后操作系统自己回收;所以在用完这些空间之后要free去释放这些空间,不然如果程序一直运行下去会一直消耗内存;

4.常见的动态内存错误

1.对NULL指针解引用

void test()
{
	int* p = (int*)malloc(INT_MAX / 4);
	*p = 20;//如果p的值是NULL,就会有问题
	free(p);
}
int main()
{
	test();
	return 0;
}

这里是很明显的没有判断函数的返回值,如果函数开辟空间失败的话,我们就是在对空指针进行解引用;

2.对动态内存开辟的越界访问

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

3.对非动态开辟内存进行释放

void test()
{
	int a = 10;
	int* p = &a;
	free(p);//ok?
}

 这是一定不可以的,free只能释放堆区的内存空间,不能释放在栈区上的空间;

4.使用free释放内存开辟的一部分

void test()
{
	int* p = (int*)malloc(100);
	p++;
	free(p);//p不再指向动态内存的起始位置
}

这也是不行的,free从起始位置开始释放空间;

5.对同一块内存空间重复释放

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

这里我们释放了那一块空间但是p并没有被置为NULL,所以对p而言其实只是单纯有一个地址,但是那块空间已经不属于你了;

6.动态内存开辟忘记释放(内存泄露)

void test()
{
	int* p = (int*)malloc(100);
	if (NULL != p)
	{
		*p = 20;
	}
}
int main()
{
	test();
	while (1);
}

我们在一个函数动态开辟了一块空间,出了函数这块空间并不会自己销毁,但是指针p是一个局部变量,出了作用域就销毁了,这时候我们就再也找不到这块空间的起始地址了,下面还有一个死循环,当程序一直不结束,这就导致了内存泄露的问题;

切记:动态开辟的空间⼀定要释放,并且正确释放。

5.动态内存的经典笔试题目

题目1

void GetMemory(char* p)
{
	p = (char*)malloc(100);
}
void Test(void)
{
	char* str = NULL;
	GetMemory(str);
	strcpy(str, "hello world");
	printf(str);
}

题目解析:

同时也有内存泄漏的问题;

题目2

char* GetMemory(void)
{
	char p[] = "hello world";
	return p;
}
void Test(void)
{
	char* str = NULL;
	str = GetMemory();
	printf(str);
}

题目解析:

题目3

void GetMemory(char** p, int num)
{
	*p = (char*)malloc(num);
}
void Test(void)
{
	char* str = NULL;
	GetMemory(&str, 100);
	strcpy(str, "hello");
	printf(str);
}

这个的错误是没有及时释放动态开辟内存空间,假设程序一直运行,会导致内存泄漏的问题;

题目4

void Test(void)
{
	char* str = (char*)malloc(100);
	strcpy(str, "hello");
	free(str);
	if (str != NULL)
	{
		strcpy(str, "world");
		printf(str);
	}
}

题目解析:

6.柔性数组

柔性数组是定义在结构体中,它的大小是不确定的;所以对于柔性数组来说,要使用动态内存给它分配空间;

语法形式:

struct stu
{
	int a;
	int arr[];//这就是柔性数组;
};

//有的编译器可能不支持上面柔性数组的形式
struct stu
{
	int a;
	int arr[0];//这种形式也是柔性数组,大小是不确定的;
};

有几个注意的点:

柔性数组是必须是结构体的最后一个成员;

柔性数组的前面至少有一个成员;

结构体大小不包括柔性数组的大小;

//结构体的大小不包括柔性数组的大小
struct stu
{
	int a;
	int arr[];
};

int main()
{
	printf("%zd", sizeof(struct stu));
	return 0;
}

同时,有柔性数组的结构体也是存在内存对齐的;

柔性数组的使用

int main()
{
	struct stu *ps = (struct stu*)malloc(sizeof(struct stu) + 10 * sizeof(int));
	if (ps == NULL)
	{
		return 1;
	}
	ps->a = 100;
	ps->ch = 'w';
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		ps->arr[i] = i;
	}
	free(ps);
	ps = NULL;
	return 0;
}

7.C/C++的内存区域划分

C/C++程序内存分配的⼏个区域:
1. 栈区(stack):在执⾏函数时,函数内局部变量的存储单元都可以在栈上创建,函数执⾏结束时这些存储单元⾃动被释放。栈内存分配运算内置于处理器的指令集中,效率很⾼,但是分配的内存容量有限。 栈区主要存放运⾏函数⽽分配的局部变量、函数参数、返回数据、返回地址等。
2. 堆区(heap):⼀般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配⽅式类似于链表。
3. 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
4. 代码段:存放函数体(类成员函数和全局函数)的⼆进制代码

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

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

相关文章

【每日一题】回旋镖的数量

文章目录 Tag题目来源解题思路方法一&#xff1a;组合数学 写在最后 Tag 【组合数学】【数组】【2024-01-08】 题目来源 447. 回旋镖的数量 解题思路 方法一&#xff1a;组合数学 思路 以数组 points 中的每一个为回旋镖的中心 i&#xff0c;在数组 points 中找距离中心 i…

leetcode:滑动窗口

目录 1.定长滑动窗口 1.1 几乎唯一子数组的最大和(使用map来计数) 1.2 长度为k子数组中的最大和 2.不定长滑动窗口 2.1 最多k个重复元素的最长子数组 2.2 绝对差不超过限制的最长连续子数组(multiset&#xff09; 2.3 将x减到0的最小操作数(正难则反 逆向思维) 2.4 统计…

Windows Server 2003 (NT 5.2.3790.0) 构建指南

Windows Server 2003 (NT 5.2.3790.0) 构建指南 版本 10b&#xff0c;最后更新于 2021/10/21 原文 https://rentry.co/build-win2k3 指令在 XP SP3 x86、Win7 SP1 x86/x64 和 Win10 x64 下测试&#xff0c;结果在其他操作系统下可能会有所不同。 该指南由一位无名的匿名人士维…

每日算法打卡:子矩阵的和 day 8

文章目录 原题链接题目描述输入格式输出格式数据范围输入样例&#xff1a;输出样例&#xff1a; 题目分析示例代码 原题链接 796. 子矩阵的和 题目难度&#xff1a;简单 题目描述 输入一个 n 行 m 列的整数矩阵&#xff0c;再输入 q 个询问&#xff0c;每个询问包含四个整数…

6.综合案例

1. 需求描述 1.1 显示所有员工信息 URI:emps 请求方式:GET 显示效果 1.2 添加操作- 去往添加页面 显示添加页面: URI:emp 请求方式:GET 显示效果 1.3 添加操作- 添加员工 添加员工信息: URI:emp 请求方式:POST 显示效果:完成添加, 重定向到 list 页面。 1.4…

.NET国产化改造探索(四)、银河麒麟安装Nginx

随着时代的发展以及近年来信创工作和…废话就不多说了&#xff0c;这个系列就是为.NET遇到国产化需求的一个闭坑系列。接下来&#xff0c;看操作。 上一篇介绍了如何在银河麒麟操作系统上安装.NET运行环境&#xff0c;这篇文章详细介绍下在银河麒麟操作系统上安装Nginx。 安装…

Unity C# 枚举多选

枚举多选 &#x1f96a;例子&#x1f354;判断 &#x1f96a;例子 [System.Flags]public enum TestEnum{ None 0,Rooms 1 << 1,Walls1<<2,Objects1<<3,Slabs 1 << 4,All Rooms|Walls|Objects|Slabs}&#x1f354;判断 TestEnum test TestEnum.R…

全网最全Midjourney以图生图的详细教程 内有6种案例 小白必收藏!!!!

手把手教你入门绘图超强的AI绘画程序&#xff0c;用户只需要输入一段图片的文字描述&#xff0c;即可生成精美的绘画。给大家带来了全新保姆级教程资料包&#xff08;文末可获取&#xff09; 基础介绍 本篇文章&#xff0c;将介绍如何利用Midjourney完成图生图的方式&#xf…

Xfs文件系统磁盘布局

目录 一&#xff0c;CentOS下Xfs文件系统的安装 二&#xff0c;准备工作 三&#xff0c;AG结构 四&#xff0c;AG超级块 五&#xff0c;AG空闲磁盘空间管理 六&#xff0c;ABTB的Btree 七&#xff0c;ABTB/ABTC的节点块管理 八&#xff0c;inode节点管理 九&#xff0…

LINUX内核故障问题之SKB DROP

关键词 Red Hat Enterprise Linux (RHEL) 7.6SKB linearization failedvm.min_free_kbytes 一、问题现象 一台业务主机dmesg 日志中频繁有以下报错&#xff1a; [qede_ start_ xmit :1289(p1p2)]SKB linearization failed - silently dropping this SKB。 二、问题分析 1…

爆肝整理,性能测试-交易系统升级压测思路,一篇不走弯路...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 交易系统性能是体…

【性能测试】JMeter分布式测试及其详细步骤

性能测试概要 性能测试是软件测试中的一种&#xff0c;它可以衡量系统的稳定性、扩展性、可靠性、速度和资源使用。它可以发现性能瓶颈&#xff0c;确保能满足业务需求。很多系统都需要做性能测试&#xff0c;如Web应用、数据库和操作系统等。 性能测试种类非常多&#xff0c…

QT应用篇:QT自定义最小化托盘显示和操作

将应用程序最小化到托盘任务栏中,可以使用Qt框架中的QSystemTrayIcon类。该类允许应用程序在关闭窗口后最小化到系统托盘,保持在后台运行,同时可以显示应用程序图标、添加右键菜单功能以及发送消息通知等。通过学习这些技术,能够为自己的Qt应用程序增加更多的交互性和便利性…

C++ TinyWebserver 部署到Linux下,并运行(使用的是Vmware的虚拟机运行Ubuntu20.04)

环境&#xff1a;VmwareUbuntu20.04 1. Tinyweb server项目地址&#xff1a;https://github.com/qinguoyi/TinyWebServer 2. 首先进行mysql5.7的安装&#xff1a; 参考教程 &#xff1a; Ubuntu20.04安装MySQL5.7-实测3种方法&#xff08;保姆级教程&#xff09;&#xff1a;…

《软件项目接口安全设计规范》

1.token授权机制 2.https传输加密 3.接口调用防滥用 4.日志审计里监控 5.开发测试环境隔离&#xff0c;脱敏处理 6.数据库运维监控审计 软件全套文档&#xff1a;软件开发全套资料-CSDN博客

Python武器库开发-武器库篇之C段扫描器开发(四十三)

Python武器库开发-武器库篇之C段扫描器开发(四十三) 在我们进行渗透过程中的信息收集的步骤时&#xff0c;收集资产目标的C段也是非常重要的一部分。 C段是指互联网中的一类IP地址。IP地址是互联网上每台设备的唯一标识符。IP地址由一系列数字组成&#xff0c;通常以点分十进…

Pytorch种torch.cat与torch.stack的区别

torch.cat 和 torch.stack 是 PyTorch 中用于拼接张量的两个不同的函数&#xff0c;它们的主要区别在于拼接的方式和创建的维度。 torch.cat&#xff1a; 拼接方式&#xff1a; torch.cat 是按照给定的维度&#xff08;dim 参数&#xff09;将多个张量沿着该维度拼接。在拼接的…

科技云报道:“存算一体”是大模型AI芯片的破局关键?

科技云报道原创。 在AI发展历史上&#xff0c;曾有两次“圣杯时刻”。 第一次发生在2012年10月&#xff0c;卷积神经网络&#xff08;CNN&#xff09;算法凭借比人眼识别更低的错误率&#xff0c;打开了计算机视觉的应用盛世。 第二次是2016年3月&#xff0c;DeepMind研发的…

Python 面向对象知识点补充

Python 面向对象知识点补充 【一】Mixins机制 【1】概念 Mixins&#xff1a;是一种在面向对象编程中&#xff0c;通过组合多个类的特称来创建一个新类的技术核心机制&#xff1a;就是在多继承的背景下尽可能地提升多继承的可读性通过命名规范来满足人的思维习惯&#xff08;…

【微机原理与接口技术】期末模拟卷(2)

有不会的题可以后台问我的哦&#xff0c;看见了就会回。 本文章主要是微机的模拟卷&#xff0c;最后祝大家期末心想事成 1、微处理器为8086数据总线和地址总线为 ()位 A.16 16 B.16 32 C.16 20 D.32 32 8086是16位寄存器&#xff0c;即需要16位数据线 2、微型计算机硬件系…