C语言——动态内存管理

大家好,我是残念,希望在你看完之后,能对你有所帮助,有什么不足请指正!共同学习交流
本文由:残念ing原创CSDN首发,如需要转载请通知
个人主页:残念ing-CSDN博客,欢迎各位→点赞👍 + 收藏⭐️ + 留言📝
📣系列专栏:残念ing 的C语言系列专栏——CSDN博客

-----------------------------------------------------------CSDN-------------------------------------------------------------

目录

前言:

1. malloc 和free(  )

1.1 malloc

1.2 free

2. calloc 和 realloc

2.1 calloc 

2.2 realloc

3. 常见的动态内存的错误

3.1 对NULL指针的解引用操作

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

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

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

3.5 对同一块动态内存多次释放

3.6 动态开辟内存忘记释放(内存泄漏)

4. 动态内存经典笔试题分析

题目1:

题目2:

题目3:

题目4:

5. 柔性数组

5.1 柔性数组的特点

5.2 关于柔性数组的使用

5.3 柔性数组的优点

6. C/C++中程序内存区域划分


前言:

为什么要有动态内存管理???

我们平时在开辟空间时是这样的:

int a=10;//在栈空间上开辟4个字节
char arr[10]={0};//在栈空间上开辟10个字节的连续空间

这样开辟的空间有两个弊端:

1、空间开辟是固定不变的

2、数组在申明的时候,必须指定数组的长度,数组空间一旦确定了大小不能调整

我们会遇到很多情况,有时候我们需要的空间大小在程序运行的时候才知道,这时开劈好的空间会不能满足,或者开辟多了,会导致大量的空间浪费。

这就有了动态内存开辟,让程序员可以自己申请需要的空间和释放不需要的空间,这就比较灵活了。

1. malloc 和free( <stdlib.h> )

1.1 malloc

这是一个动态内存开辟的函数:

void* malloc(size_t size);

 作用:可以向内存申请一块连续可用的空间,并返回指向这块空间的指针。

注意:如果开辟成功,则返回一个指针开辟好空间的指针。

           如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做判断。

           返回值的指针类型是void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。

            如果参数size 为0,malloc的行为标准是未定义的,取决于编译器。

补充:malloc 申请的空间是在内存的堆区的

malloc的使用:

1.2 free

C语言还为我们提供了一个专门用来做动态内存的释放和回收的函数 free :

void free (void* ptr);

作用:用来释放动态开辟的内存。

注意:1、如果参数 ptr 指向的空间不是动态开辟的,那么 free 函数的行为是未定义的。

            2、如果参数 ptr 是NULL指针,则函数什么都不做。

补充:free 只是会把 ptr 指向的空间释放了,但是 ptr 里面还存着那块空间的起始地址,我们为了防止 ptr 后面会成为野指针,我们必须要把 ptr 制为空指针。

free 的使用:

2. calloc 和 realloc

2.1 calloc 

除了malloc外 还有一个 calloc 函数,它也是用来动态内存开辟的:

void* calloc(size_t num,size_t size);

注意:函数的功能是 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始为0。

           与函数malloc的区别只在于calloc会返回地址之前把申请的空间的每个字节初始化为全0。

int main()
{
     int *p = (int*)calloc(10, sizeof(int));
     if(NULL != p)
     {
         int i = 0;
         for(i=0; i<10; i++)
         {
             printf("%d ", *(p+i));
         }
     }
     free(p);
     p = NULL;
     return 0;
}

作用:对申请的内存空间进行初始化。

2.2 realloc

 有时候我们会发现之前申请的空间太小了,有时候我们会觉得申请的空间过大了,为了合理的使用内存,我们会对内存的大小作合理的调整,realloc 函数就可以做到对动态开辟内存大小的调整。

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

ptr :要调整的内存地址。

size:调整之后新大小。

注意:返回值为调整之后的内存的起始位置。

           这个函数调整原内存空间大小的基础上,还会将原来内存的数据移动到新的空间。

realloc 在调整内存空间的说存在两种情况:

情况1:在已经开辟好的空间后边,没有足够的空间,直接进行空间的扩大。在这种情况下,realloc 函数会在内存的堆区重新找一个空间(满足新的空间的大小需求的)同时会把旧的数据拷贝到新的空间,然后释放旧的空间,同时返回新的空间的起始地址。

情况2:在已经开辟好的空间后面,有足够的空间,直接进行扩大。扩大空间后,直接返回旧的空间的起始地址。

补充:我们不能直接把realloc的返回值放到目标指针中,因为如果申请失败,那之前的空间也会找不到。我们一般要先将realloc函数的返回值放在一个临时指针中,判断是不是NULL指针,再放到目标指针中。

举例:

#include <stdio.h>
#include <stdlib.h>
int main()
{
	int* ptr = (int*)malloc(100);
	if (ptr != NULL)
	{
		perror("malloc");
	}
	else
	{
		return 1;
	}
	//扩展容量

	int* p = (int*)realloc(ptr, 1000);
	if (p != NULL)
	{
		ptr = p;
	}
	else
	{
		perror("realloc");
	}
	//业务处理
	free(ptr);
	return 0;
}

3. 常见的动态内存的错误

3.1 对NULL指针的解引用操作

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

如果一定要这样的话:就必须先判断一下p是不是NULL指针。

3.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.3 对非动态开辟内存使用free释放

void test()
{
	int a = 10;
	int* p = &a;
	free(p);//p指向的空间不是在堆区上
}

补充:malloc、calloc、 realloc 申请的空间,如果不主动释放,出了作业域是不会销毁的

释放的方式:1、free主动释放        2、直到程序结束,才由操作系统回收

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

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

3.5 对同一块动态内存多次释放

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

3.6 动态开辟内存忘记释放(内存泄漏)

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

忘记释放不再使⽤的动态开辟的空间会造成内存泄漏。 切记:动态开辟的空间⼀定要释放,并且正确释放。

4. 动态内存经典笔试题分析

题目1:

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

出错的点:

1、GetMemory函数采用值传递的方式,无法将malloc 开辟空间的地址,返回放在str 中,调用结束后str 依然是NULL指针。

2、strcpy 中使用了str,就是对NULL指针解引用操作,程序崩溃。

3、内存泄漏。

解决方法:

1、

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

2、

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

题目2:

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

出错的点:在出GetMemory函数后,存储 hello  world 的空间被回收了,就不能在使用了,str 也成为了野指针。(放回栈空间地址的问题)如果要改,只能在数组前加 static。

题目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);
	}
}

出错的点: free 后str为野指针了。再访问就非法访问了。

5. 柔性数组

在我们的认识范围内,也许我们从来没有听说过柔性数组这个概念,但是它确实是存在的。

在结构体中的最后一个元素允许是未知大小的数组,这就叫做 柔性数组 成员。

struct st_type
{
     int i;
     int a[0];//柔性数组成员
};
struct st_type
{
     int i;
     int a[];//柔性数组成员
};

5.1 柔性数组的特点

1、结构体中的柔性数组成员前面至少一个其他成员。比如:int char等一些变量

2、sizeof 返回这种结构体的大小不包括柔性数组的内存。

3、包含柔性数组成员的结构体用malloc() 函数进行内存的分配,并且分配的内存一个大于结构体的大小,一适应柔性数组的预期大小。

struct st
{
	int i;
	int a[0];//柔性数组成员
};
int main()
{
	printf("%d\n", sizeof(struct st));//输出的是4
	return 0;
}

5.2 关于柔性数组的使用

//代码1
#include <stdio.h>
#include <stdlib.h>
struct st
{
	int i;
	int a[0];//柔性数组成员
};
int main()
{
	struct st* p = (struct st*)malloc(sizeof(struct st) + 10 * sizeof(int));//为结构体分配空间+为柔性数组分配空间
	p->i = 30;
	for (int i = 0; i < 10; i++)
	{
		p->a[i] = i;
	}
	//分配的空间不足时
	struct st* ptr = (struct st*)realloc(p, sizeof(struct st) + 20 * sizeof(int));
	if (ptr == NULL)
	{
		perror("realloc");
		return -1;
	}
	else
	{
		p = ptr;
	}
	//再使用
	for (int i = 10; i < 20; i++)
	{
		p->a[i] = i;
	}
	for (int i = 0; i < 20; i++)
	{
		printf("%d ", p->a[i]);
	}
	printf("\n%d\n", p->i);

	//销毁
	free(p);
	p = NULL;
	return 0;
}

5.3 柔性数组的优点

我们先看一下下面的代码,这个结构体的设计也可以完成同样的效果。

#include <stdio.h>
#include <stdlib.h>
struct st2
{
	int i;
	int* p_a;
};
int main()
{
	struct st2* pt=(struct str2*)malloc(sizeof(struct st2));
	pt->i = 10;
	pt->p_a=malloc(sizeof(int) * 10);//单独让其指向一个分配好的空间
	//使用
	for (int i = 0; i < 10; i++)
	{
		pt->p_a[i] = i;
	}
	//空间不足
	int* ptr=(int*)realloc(pt->p_a, sizeof(int) * 20);
	if (ptr == NULL)
	{
		perror("raslloc");
		return -1;
	}
	else
	{
		pt->p_a = ptr;
	}
	//再使用
	for (int i = 10; i < 20; i++)
	{
		pt->p_a[i] = i;
	}
	for (int i = 10; i < 20; i++)
	{
		printf("%d ", pt->p_a[i]);
	}
	return 0;
}

上述代码其实完成了同样的功能,但是代码1的实现更突出且有两个好处:

第一个好处是:方便内存释放

如果我们的代码是在一个给别人用的函数中,只能里面做了两次内存分配,并把整个结构体返回给用户。用户调用free 可以释放结构体,但是用户并不知道这个结构体的成员也需要 free,所以你不能指望用户来发现这件事。所以,如果我们把结构体的内存以及成员一次性分配好内存,并且返回给用户一个结构体指针,用户做一次 free 就可以把所有内存都释放掉了。

第二个好处是:有利于访问速度

连续的承诺有益于提高访问速度,也有益于减少内存碎片(少几个中间商)

6. C/C++中程序内存区域划分

C/C++程序内存分配的⼏个区域:

1. 栈区(stack):在执⾏函数时,函数内局部变量的存储单元都可以在栈上创建,函数执⾏结束时 这些存储单元⾃动被释放。栈内存分配运算内置于处理器的指令集中,效率很⾼,但是分配的内 存容量有限。 栈区主要存放运⾏函数⽽分配的局部变量、函数参数、返回数据、返回地址等。 2. 堆区(heap):⼀般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配⽅ 式类似于链表。

3. 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。

4. 代码段:存放函数体(类成员函数和全局函数)的⼆进制代码。

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

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

相关文章

如何在前端项目里接入Sentry监控系统并通过企业微信通知

能不能让用户录个屏过来呀&#xff1f; 用户使用的是什么机型的手机&#xff1f; 用户使用的什么浏览器呀&#xff1f; 用户的网络是什么情况&#xff1f; … … 线上出现问题时&#xff0c;技术部和业务部同学之间的对话诸如此类…业务同学也很栓Q呀&#xff0c;硬着头皮去问客…

一篇文章带你了解C++中隐含的this指针

文章目录 一、this指针的引出二、this指针的特性【面试题】 一、this指针的引出 我们先来定义一个日期类Date&#xff0c;下面这段代码执行的结果是什么呢&#xff1f; class Date { public:void Init(int year, int month, int day){_year year;_month month;_day day;}v…

STM32+ESP8266 实现物联网设备节点

一、硬件准备 本设备利用STM32F103ZE和ESP8266实现了一个基本的物联网节点&#xff0c;所需硬件如下 1、STM32F103ZE开发板 2、ESP8266模组&#xff08;uart接口&#xff09; 3、ST-LINK&#xff08;下载用&#xff09; 4、USB转串口模块&#xff08;调试用&#xff09; 二…

mac裁剪图片

今天第一次用mac裁剪图片&#xff0c;记录一下过程&#xff0c;差点我还以为我要下载photoshop了&#xff0c; 首先准备好图片 裁剪的目的是把图片的标题给去掉&#xff0c;但是不能降低分辨率&#xff0c;否则直接截图就可以了 解决办法 打开原始图片(不要使用预览&#xf…

利用柯西积分公式证明最大模定理

一、利用柯西积分公式证明最大模定理 一、利用柯西积分公式证明最大模定理 设复变函数f(z)在封闭区域上的解析&#xff0c;则该复变函数的模|f(z)|的最大值只能出现在该区域的边界上&#xff0c;除非是个常数

容器化部署 Jenkins,并配置SSH远程操作服务器

目录 一、Jenkins是什么 二、常见的部署Jenkins的方法 三、为什么选择容器化部署 四、容器化部署Jenkins步骤 1、安装 Docker 2、获取 Jenkins 镜像 3、创建并运行容器 4、访问 Jenkins 4.1 查看初始密码问题 5、配置 Jenkins 5.1 安装插件 5.2 创建管理员用户 5.3…

兄弟HL-1208黑白激光打印机清零方法

兄弟HL-1208黑白激光打印机基本参数&#xff1a; 产品类型&#xff1a;黑白激光打印机&#xff08;定位类型家用&#xff09; 最大打印幅面&#xff1a;A4 最高分辨率&#xff1a;600600dpi 黑白打印速度&#xff1a;20ppm 内存标配&#xff1a;1MB&#xff0c;最大&#…

数据结构(顺序表)

文章目录 一、线性表1、线性表1.1、线性表的定义1.2、线性表的操作 2、顺序表2.1、顺序表的实现--静态分配2.2、顺序表的实现--动态分配2.2、顺序表的特点 3、顺序表的基本操作3.1、插入操作3.2、删除操作3.3、查找操作3.2、按位查找3.2、按值查找 一、线性表 1、线性表 1.1、…

小白学python,都需要准备什么?有什么好的学习规划么?

对于一个0基础的小白来说&#xff0c;学习Python需要准备以下内容&#xff1a; 确定学习目标&#xff1a;在开始学习Python之前&#xff0c;你需要明确自己的学习目标。你是想学习Python的基础语法、数据结构、面向对象编程、Web开发、数据分析还是机器学习等方面的内容&#…

Ultraleap 3Di示例Interactable Objects组件分析

该示例代码位置如下&#xff1a; 分析如下&#xff1a; Hover Enabled&#xff1a;悬停功能&#xff0c;手放在这个模型上&#xff0c;会触发我们手放在这个模型上的悬停功能。此时当手靠近模型的时候&#xff0c;手的模型的颜色会发生改变&#xff0c;反之&#xff0c;则不会…

蓝凌OA sysUiExtend.do 任意文件上传漏洞复现

0x01 产品简介 蓝凌核心产品EKP平台定位为新一代数字化生态OA平台,数字化向纵深发展,正加速构建产业互联网,对企业协作能力提出更高要求,蓝凌新一代生态型OA平台能够支撑办公数字化、管理智能化、应用平台化、组织生态化,赋能大中型组织更高效的内外协作与管理,支撑商业…

[UI5 常用控件] 02.Title,Link,Label

文章目录 前言1. Title1.1 结合Panel1.2 结合Table1.3 Title里嵌套Link 2. Link3. Label3.1 普通用法3.2 在Form里使用 前言 本章节记录常用控件Title,Link,Label。 其路径分别是&#xff1a; sap.m.Titlesap.m.Linksap.m.Label 1. Title Title可以结合其他控件一起使用 1.…

Android源码设计模式解析与实战第2版笔记(四)

第三章 自由扩展你的项目–Builder 模式 Builder 模式的定义 将一个复杂对象的构建与它的表示分离&#xff0c;使得同样的构建过程可以创建不同的表示。 Builder 模式的使用场景 相同的方法&#xff0c;不同的执行顺序&#xff0c;产生不同的事件结果时 多个部件或零件&…

使用Spring Boot和Tess4J实现本地与远程图片的文字识别

概要&#xff1a; 在本文中&#xff0c;我们将探讨如何在Spring Boot应用程序里集成Tess4J来实现OCR&#xff08;光学字符识别&#xff09;&#xff0c;以识别出本地和远程图片中的文字。我们将从添加依赖说起&#xff0c;然后创建服务类以实现OCR&#xff0c;最后展示如何处理…

Redis客户端之Jedis(一)介绍

目录 一、Jedis介绍&#xff1a; 1、背景&#xff1a; 2、Jedis连接池介绍&#xff1a; 二、Jedis API&#xff1a; 1、连接池API 2、其他常用API&#xff1a; 三、SpringBoot集成Jedis&#xff1a; 1、Redis集群模式&#xff1a; &#xff08;1&#xff09;配置文件…

MySql8的简单使用(1.模糊查询 2.group by 分组 having过滤 3.JSON字段的实践)

MySql8的简单使用&#xff08;1.模糊查询 2.group by 分组 having过滤 3.JSON字段的实践&#xff09; 一.like模糊查询、group by 分组 having 过滤 建表语句 create table student(id int PRIMARY KEY,name char(10),age int,sex char(5)); alter table student add height…

TCP 状态转换以及半关闭

TCP 状态转换&#xff1a; 上图中还没有进行握手的时候状态是关闭的。 三次握手状态的改变&#xff1a; 客户端发起握手。 调用 connect() 函数时状态转化为&#xff1a;SYN_SENT。调用 listen() 函数时状态转换为&#xff1a;LISTEN。ESTABLISHED是被连接的状态。 四次挥手…

卢禹舜个展开幕作品震撼引人驻足

——“天地人和•大道不孤——卢禹舜中国画作品展”在贵州美术馆盛大开展 1月25日&#xff0c;寒风料峭&#xff0c;冬意正浓&#xff0c;但贵州美术馆大厅内却人潮涌动、热闹非凡。下午3点&#xff0c;由中国国家画院、贵州省文化和旅游厅主办&#xff0c;贵州画院(贵州美术馆…

字符串和C预处理器

本文参考C Primer Plus第四章学习 文章目录 常量和预处理器const限定符 1. 常量和预处理器 有时&#xff0c;在程序中要使用常量。例如&#xff0c;可以这样计算圆的周长&#xff1a; circumference 3.14159 * diameter; 这里&#xff0c;常量3.14159 代表著名的常量 pi(π)。…

详解静态网页数据获取以及浏览器数据和网络数据交互流程

目录 前言 一、静态网页数据 二、网址通讯流程 1.DNS查询 2.建立连接 3.发送HTTP请求 4.服务器处理请求 5.服务器响应 6.渲染页面 7.页面交互 三、URL/POST/GET 1.URL 2.GET 形式 3.POST 形式 四.获取静态网页数据 前言 在网站设计领域&#xff0c;基于纯HTM…