【C/C++动态内存 or 柔性数组】——对动态内存分配以及柔性数组的概念进行详细解读(张三 or 李四)

news2024/12/28 15:04:23

在这里插入图片描述


目录

  • 前言(栈区、堆区、静态区)
  • 动态内存函数
    • malloc与free
    • calloc与free
    • realloc与free
  • 常见的动态内存错误
  • 经典笔试题(再见张三)
  • 柔性数组

前言(栈区、堆区、静态区)

请耐心看完,看完后就会对内存中的空间划分有了更深刻的认识!

我们知道,任何一个变量的创建都会向内存申请空间用来存放,而在内存中的空间又划分为几个区域、最主要划分为:栈区、堆区、静态区

而我们平常创建变量或者数组,如下:

int a=0;
int arr[1000];

这里的a与arr都是在栈区开辟空间的,栈区的特点之一就是出了作用域就会自动销毁,所以它们的生命周期只要出了所在的作用域就结束了。因此在栈区上开辟空间的变量一般都是:局部变量、形参这种

而且我们发现,在栈区上开辟空间的一些变量,它们的大小都是固定的,就比如上文的数组arr,它的大小就是固定的4000字节,但是我们可以想一下,有时候在使用它的时候,并不需要这么多的空间,可能仅仅只需要10个整形大小的空间,而后面的990个整形空间都会被浪费掉,着实是可惜呀!


那我们不禁美滋滋的会这么想象,会不会存在我们想用多少空间,就开辟多少空间的可能呢?答案是有的!
我们上面提到了内存中还划分有堆区,而堆区的特点之一就是:可以按自己的需求开辟空间,并且该空间出了作用域不会自动销毁,只能人工销毁,这就实现了我们想要的需求。

那么应如何在堆区开辟空间呢?这里就涉及到了以下讲到的几个函数:malloc、realloc、calloc,还有用来释放空间的free


可能有人还会疑问,上面的静态区是干嘛的,所谓的静态区,它的特点是:永恒存在、生命周期一直到程序结束所以在静态区开辟空间的变量一般为:常量、const修饰的常变量、全局变量、以及static修饰的静态 全局/局部 变量。
在这里插入图片描述


动态内存函数

我们上面已经讲过了,动态内存分配是在堆区完成、并且空间是由程序员自己释放,因此切记,malloc、calloc、realloc与free都是成对出现的

malloc与free

首先是malloc,向内存申请size字节的空间,然后返回该空间的起始地址。
在这里插入图片描述

使用演示

#include<stdio.h>
#include<stdlib.h>//头文件
int main()
{
	int* p = (int*)malloc(10*sizeof(int));
	//开辟10个整形大小的空间(40byte),然后用指针p来接收该空间的起始地址
	//因为p是int*类型的,所以将该空间强制类型转换成(int*),保证用来接收的指针类型与开辟空间的类型一致
	if (p == NULL)
	{
		perror("malloc");
		return 1;
	}
	//对空间进行一个判断,假如开辟失败,打印错误,并返回。return 1表示非正常返回、
	//开辟成功正常使用
	//...
	free(p);//使用完一定记得释放!(从哪里申请,从哪里释放,后面会将注意事项)
	p = NULL;//将指针置空
	return 0;
}

这里一定要对p进行判断,因为假如空间开辟失败,p就是一个空指针,后面假如对p进行操作与使用,很可能会出现很大的问题!

calloc与free

calloc与malloc很像,使用也基本相同,只不过它是这样使用的:开辟num个大小为size的空间,并且将空间的每个字节都初始化为0,而malloc开辟的空间里面的值是随机值。
在这里插入图片描述

使用演示

#include<stdio.h>
#include<stdlib.h>//头文件
int main()
{
	int* p = (int*)calloc(10,sizof(int));
	//开辟个10个空间,每个空间大小为一个整形大小(一共40byte),然后用指针p来接收该空间的起始地址
	//因为p是int*类型的,所以将该空间强制类型转换成(int*),保证用来接收的指针类型与开辟空间的类型一致
	if (p == NULL)
	{
		perror("calloc");
		return 1;
	}
	//对空间进行一个判断,假如开辟失败,打印错误,并返回。return 1表示非正常返回、
	//开辟成功正常使用
	//...
	free(p);//使用完一定记得释放!(从哪里申请,从哪里释放,后面会将注意事项)
	p = NULL;//将指针置空
	return 0;
}

realloc与free

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

但是会存在原地扩容和异地扩容两种情况
在这里插入图片描述
使用演示

#include<stdlib.h>//头文件
#include<stdio.h>

int main()
{
	int* p = (int*)malloc(40);//开辟40byte
	//判断是否开辟成功
	if (p == NULL)
	{
		perror("malloc fail");
		return 1;
	}
	//使用p指向的空间
	for (int i = 0; i < 10; i++)
	{
		*(p + i) = i;
	}
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", *(p + i));
	}
	//0 1 2 3 4 5 6 7 8 9
	//扩容,用ptr接收新空间起始地址
	int* ptr = (int*)realloc(p, 80);
	if (ptr != NULL)
	{
		//扩容成功后,让p指向ptr,ptr置空
		p = ptr;
		ptr = NULL;
	}
		//使用
		for (int i = 0; i < 20; i++)
		{
		*(p + i) = i;
	}
	for (int i = 0; i < 20; i++)
	{
		printf("%d ", *(p + i));
	}
	//使用完释放
	free(p);
	p = NULL;
	return 0;
}

常见的动态内存错误

我们在使用动态内存分配时总是难免会犯一些不必要的错误,毕竟人非圣贤,孰能无过,接下来我将列举这些常见的错误,以警示避免!
1、对空指针的解引用
例:

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

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的时候越界访问,error!
 }
 free(p);
}

3、对非动态开辟内存使用free释放
例:

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

4、使用free释放一块动态开辟内存的一部分
例:

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

5、 对同一块动态内存多次释放
例:

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

6、 动态开辟内存忘记释放(内存泄漏)
例:

void test()
{
 int *p = (int *)malloc(100);
 if(NULL != p)
 {
 *p = 20;
 }
 //忘记释放!error!
}
int main()
{
 test();
 while(1);
}

动态开辟的内存空间不使用的时候一定要记得释放!


经典笔试题(再见张三)

接下来通过一些经典笔试题的讲解来加深对动态内存分配的理解:
题目一:解释运行Test函数出现的结果

void GetMemory(char *p)
{
p = (char *)malloc(100);
}
void Test(void)
{
char *str = NULL;
GetMemory(str, 100);
strcpy(str, “hello”);
printf(str);
}
在这里插入图片描述

分析:

在这里,str首先置空,把str传过去,用指针p来接收,然后p再指向新开辟的空间,再把hello拷贝到该空间,接着打印。
听起来好像没什么毛病,但是我们忽略了以下几点!首先,malloc开辟的空间并没有free,造成内存泄漏,这时最明显的错误!
然后,GetMemory这里只是传址调用,也就是说,p确实指向了那块空间,但是实际上str并没有指向,这里只是把str=NULL的值,传了过去,p=NULL,然后对p进行操作,我们知道,传值调用,形参的改变不会影响实参!所以str仍是NULL,而strcpy一个空指针,就涉及到了对空指针的解引用,ERROR!
这两处错误最为致命!

作为修改,我们可以这样改正:

void GetMemory(char** p)//一级指针的地址用二级指针来接收
{
	*p = (char*)malloc(100);//*p 等价于*&str,等价于str,即str=......
}
void Test(void)
{
	char* str = NULL;
	GetMemory(&str);//传址调用,对形参的修改会影响实参
	strcpy(str, "hello");
	printf(str);
	free(str);//释放
	str = NULL;
}

笔试题二:以下代码运行结果:

#include<stdlib.h>
#include<stdio.h>
#include<string.h>
char* Getmemory()
{
char p[] = “hello world!”;
return p;
}
void test()
{
char* str = NULL;
str = Getmemory();
printf(str);
}
int main()
{
test();
return 0;
}

看起来没什么问题,str来接收Getmemory返回的地址,然后打印,按理来说应该是hello world!,但是,真实结果却是一堆乱码
在这里插入图片描述
这是为什么呢?
分析:

在前言那块,讲到了栈区的特点就是出作用域后会自动销毁,我们看这里的p,p是数组名,表示数组首元素的地址,在这里即字符’h‘的地址,然后返回该地址用str来接收,但是!别忘记p是个局部变量,局部变量在栈区开辟空间,出作用域后会自动销毁!也就是说
虽然传给了h所在的地址,但是当它传过去的那一刻,p所在的空间就自动销毁,而str依然记着那块空间,但是此时的那块空间已经不属于p了,这就造成了野指针的访问,谁也不知道那块空间销毁后里面是什么,所以打印乱码。

图解:
在这里插入图片描述

张三 and 李四

举个张三与李四的故事来方便大家理解(🤭),张三与李四是一对情侣,这一天,好不容易放假,李四就去酒店开了个房,然后把房间号告诉了李四,然后自己在房间里等他,等啊等,发现张三一直在打lol,还没有过来,就气的也没有告诉张三,就把房间退了,然后房间里现在住着王二麻子和他对象,假如张三再去找李四,确实,他记得房间号码,也能找到位置,但是他不知道房间里已经换人了,此时他不打开房间还好,假如打开的话…(狗头保命🐕)

这里的张三就好比题目里的str,p->酒店房间,p的地址->房间号码,空间销毁之前里面住的是李四,销毁后住的王二麻子。
str虽然能找到p之前指向的空间,但空间里的内容早已换了~

柔性数组

定义

柔性数组这个名词听起来很高大上,但其实并没有什么特殊的,那么它是什么呢?简单来说,就是结构体中的最后一位成员为数组,并且大小未知。
举个栗子:

typedef struct st_type
{
 int i;
 int a[];//柔性数组成员,也可以写成int a[0]
}type_a;

这里的数组a,是结构体最后一位成员,并且大小未知,所以它就是所谓的柔性数组.

特点
1、结构中的柔性数组成员前面必须至少一个其他成员。
2、sizeof 返回的这种结构大小不包括柔性数组的内存。
3、包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。

使用

#include<stdlib.h>
#include<stdio.h>
struct S
{
	int i;
	int a[];
};

int main()
{
	struct S* p = (struct S*)malloc(sizeof(struct S) + 100 * sizeof(int));//为柔性数组a提供100个整形空间
	if (p == NULL)
	{
		perror("malloc fail");
		return 1;
	}
	p->i = 100;
	for (int j = 0; j < 100; j++)
	{
		p->a[j] = j;
	}

	free(p);//释放
	p = NULL;
	return 0;
}

在这里插入图片描述
这样有什么好处呢?我个人觉得,首先这个柔性数组它的空间可以按照自己的需要来开辟,不会造成大量的空间浪费,还有就是方便释放,直接一次性free整个结构体指针即可。


end
生活原本沉闷,但跑起来就会有风!

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

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

相关文章

天玑9200领跑背后,高端芯片掀起蝴蝶效应

过去一年&#xff0c;天玑9000让联发科在高端旗舰机市场掀起了一场“旋风”。全球知名市场调研机构Counterpoint Research发布的报告显示&#xff0c;联发科在全球和中国智能手机芯片市场份额中连续八个季度保持第一的领先地位&#xff0c;且在高端手机市场的份额有显著增长。另…

PowerShell 美化(谁不想要一个好看的终端呢)

PowerShell 美化安装powershellScoop 安装Oh My Posh安装字体设置应用主题花里胡哨的折腾&#xff08;bushi 多种主题任君挑选 安装powershell 地址&#xff1a;https://github.com/PowerShell/PowerShell/releases 本文主要使用 Oh My Posh 进行美化 地址&#xff1a;https…

C语言之指针(中)

目录 前言 一、字符指针 1.用法&#xff08;两种&#xff09; 2.例子 二、指针数组 三、数组指针 1.数组指针的定义 1.概念 2.例子 2.数组名 3.使用 1.使用的情景 2.例子 四、数组参数、指针参数 1.数组传参 2.指针传参 五、函数指针 1.函数的地址 2.函数指针 3.例子 4.两个特…

【C++】泛型编程之模板初阶

文章目录1. 泛型编程2. 模板2.1 模板的概念3. 函数模板3.1 函数模板概念3.2 函数模板语法3.3 函数模板的原理3.4 函数模板的实例化3.5 模板参数的匹配原则3.6 普通函数与函数模板的区别4. 类模板4.1 类模板语法4.2 类模板的实例化4.3 类模板与函数模板区别4.4 类模板分文件编写…

Word控件Spire.Doc 【文本】教程(17) ;在Word中设置文本方向

Spire.NET的Spire.是MicrosoftDoc人员对Word文档进行操作打印的.NET类库。帮助单独安装Microsoft Word在开发环境下&#xff0c;轻松便捷地创建、编辑、转换和转换Word文档。拥有近10个专业开发经验Spire系列办公文档开发打印工具&#xff0c;专注于创建、编辑、转换和Word/Exc…

C++设计模式---组合模式

文章目录使用场景组合模式的定义安全组合模式使用场景 组合模式和类与类之间的组合是不同的概念。 组合模式主要用来处理树形结构的数据&#xff0c;如果要表达的数据不是树形结构&#xff0c;就不太适合组合模式。 比如我们有一个目录结构&#xff1a; 这个目录我们把它绘…

图解LeetCode——1704. 判断字符串的两半是否相似(难度:简单)

一、题目 给你一个偶数长度的字符串 s 。将其拆分成长度相同的两半&#xff0c;前一半为 a &#xff0c;后一半为 b 。 两个字符串 相似 的前提是它们都含有相同数目的元音&#xff08;a&#xff0c;e&#xff0c;i&#xff0c;o&#xff0c;u&#xff0c;A&#xff0c;E&…

几分钟实现对恶意IP地址进行拦截,腾讯云Web防火墙实在太香了!

一、概述 在平时上网中&#xff0c;我们经常听到“xxx被拉入黑名单”、“把xxx加入白名单”&#xff0c;黑白名单成了禁止访问和允许访问的代名词&#xff0c;黑白名单是一种常见的安全机制&#xff0c;用于隔离流量&#xff0c;然后对隔离的流量采取特定操作。 黑名单代表只…

Redis高可用之持久化

一 Redis高可用 什么是高可用 在web服务器中&#xff0c;高可用是指服务器可以正常访问的时间&#xff0c;衡量的标准是在多长时间内可以提供正常服务(99.9%、99.99%、99.999%等等)。 但是在Redis语境中&#xff0c;高可用的含义似乎要宽泛一些&#xff0c;除了保证提供正常…

Pytorch实战:基于鲸鱼WOA优化1DCNN的轴承故障诊断

目录 0.引言 1.关键点 2.WOA优化1DCNN超参数实战 2.1 数据准备 2.2 1DCNN故障诊断建模 2.3 采用WOA优化1DCNN超参数 0.引言 采用1DCNN进行轴承故障诊断建模&#xff0c;并基于鲸鱼优化算法WOA对1DCNN的超参数进行优化&#xff0c;以实现更高的精度。建立一个两层的1DCNN&a…

【Transformers】第 9 章 :处理很少或没有标签

&#x1f50e;大家好&#xff0c;我是Sonhhxg_柒&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流&#x1f50e; &#x1f4dd;个人主页&#xff0d;Sonhhxg_柒的博客_CSDN博客 &#x1f4c3; &#x1f381;欢迎各位→点赞…

[C]实现能在本地存储的简易通讯录

作者&#xff1a; 华丞臧. 专栏&#xff1a;【C语言】 各位读者老爷如果觉得博主写的不错&#xff0c;请诸位多多支持(点赞收藏关注)。如果有错误的地方&#xff0c;欢迎在评论区指出。 文章目录一、文件动态通讯录二、相关代码解析2.1 初始化2.2 销毁通讯录2.3 增加联系人2.4 …

NLP模型(一)——word2vec实现

文章目录1. 整体思路2. 数据处理3. 数据准备4. 创建数据管道5. 构建模型6. 模型训练7. 加载模型得到词向量8. 总结前面我介绍了word2vec算法的两种实现算法&#xff0c;Skip−gramSkip-gramSkip−gram 以及 CBOWCBOWCBOW 算法&#xff0c;我认为理解一个算法最好的方法就是复现…

stft的窗函数设计要求和方法(COLA)

在语音处理进行短时傅里叶变换的时候&#xff0c;对窗函数是有一定要求的&#xff0c;这篇文章将对这方面的问题进行简单的阐述。 一、背景描述 常用的语音处理需要进行这样处理: stft分帧会对信号产生截断&#xff0c;为尽可能避免这种影响&#xff0c;应考虑考虑加合适的窗 …

互融云借条APP系统开发 六大系统优势全面保障

借条是指借个人或公家的现金或物品时写给对方的条子。它是一种凭证性文书&#xff0c;通常用于日常生活以及商业管理方面。借条的本质就是借款合同&#xff0c;只不过形式比较简单&#xff0c;那么电子借条也就是简单的电子借款合同。与传统的纸质合同相比&#xff0c;电子借条…

目标检测算法——YOLOv5/YOLOv7改进之结合无参注意力SimAM(涨点神器)

目录 &#xff08;一&#xff09;前言介绍 1.摘要 2.不同注意力步骤比较 &#xff08;二&#xff09;相关实验 &#xff08;三&#xff09;YOLOv5结合无参注意力SimAM 1.配置.yaml文件 2.配置common.py 3.修改yolo.py SimAM&#xff1a;无参数Attention助力分类/检测/分…

想带着学生做一个操作系统,可行性有多大?

有知乎网友提问如下: 想带着学生做一个操作系统&#xff0c;可行性有多大&#xff1f; 个人觉得可行性非常大&#xff0c;如果只是做着来玩&#xff0c;让学生了解操作系统时如何实现的话。但是&#xff0c;如果你打算今后商业化的话&#xff0c;那就另当别论了。就算你能做出来…

单片机实验——水塔自动抽水系统设计(基于Proteus仿真)

实验内容及要求 自来水供水是现代生活的一大特点&#xff0c;水塔作为储水装置是自来水系统必不可少的重要设施&#xff0c;让水塔保持一定的水量是自来水不断供的必要条件&#xff0c;本设计模拟自来水系统中水塔的自动抽水机制&#xff0c;设计分为控制系统和虚拟水塔两部分…

若依管理框架-漏洞复现

文章目录 0x00 介绍0x01 默认口令漏洞0x02 SQL注入0x03 Shiro反序列化漏洞0x04 任意文件读取/下载0x05 定时任务0x06 `swagger-ui.html`接口文档泄漏0x07 Druid未授权访问摘抄免责声明0x00 介绍 RuoYi 是一个 Java EE 企业级快速开发平台,基于经典技术组合(Spring Boot、Apa…

黑产工具情报的分析方式浅析

接下来我们以恶意爬虫、抢券工具和注册机三种工具来谈一下黑产工具情报的分析方式。 对于企业方面来说&#xff0c;黑产工具情报可以有效的提高业务安全的攻防效率。通过分析工具利用的业务接口&#xff0c;不仅可以将黑产作恶行为进行有效的追踪&#xff0c;对其进行有效的处…