【C++】---内存管理new和delete详解

news2024/10/1 17:41:43

在这里插入图片描述

一、C/C++内存分布

C/C++内存被分为6个区域:
(1) 内核空间:存放内核代码和环境变量。

(2)栈区:向下增长存放非静态局部变量,函数参数,返回值等等

(3)内存映射段:文件映射,匿名映射,动态库。

(4)堆区:向上增长用于程序运行时动态内存的分配

(5)数据段:也叫,静态区/全局域,(存放全局变量和静态变量

(6)代码段:也叫常量区(存放可读代码和只读常量)

在这里插入图片描述
看看下面代码的例题:

int globalVar = 1;
static int staticGlobalVar = 1;
void Test()
{
 static int staticVar = 1;
 int localVar = 1;
 int num1[10] = { 1, 2, 3, 4 };
 char char2[] = "abcd";
 const char* pChar3 = "abcd";
 int* ptr1 = (int*)malloc(sizeof(int) * 4);
 int* ptr2 = (int*)calloc(4, sizeof(int));
 int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
 free(ptr1);
 free(ptr3);
}
  1. 选择题:
    选项: A.栈 B.堆 C.数据段(静态区) D.代码段(常量区)
    globalVar在哪里?____ staticGlobalVar在哪里?____
    staticVar在哪里?____ localVar在哪里?____
    num1 在哪里?____

    char2在哪里?____ *char2在哪里?___
    pChar3在哪里?____ *pChar3在哪里?____
    ptr1在哪里?____ *ptr1在哪里?____

  2. 填空题:
    sizeof(num1) = ____;
    sizeof(char2) = ____; strlen(char2) = ____;
    sizeof(pChar3) = ____; strlen(pChar3) = ____;
    sizeof(ptr1) = ____;

    1. sizeof 和 strlen 区别?

1.C C C A A
2. A A A D A B
3.40 5 4 4或8 4 4或8

3.sizeof和strlen的区别:
(1)char ch2[ ]=“abcd”; // sizeof(ch2)=5 (因为末尾还有一个’\0’)
// strlen(ch2)=4 (因为strlen其他都不管,遇到’\0’就结束,‘\0’不计入
(2)sizeof()内部要看变量是否为指针,若是指针==4/8
strlen不看变量是否为指针,只看’\0’ ,遇到’\0’结束为止!!!

4.对char2,pchar3,ptr1的解释:
(1)字符串存在于常量区,当拿来初始化char2的时候,char2会在栈区开辟一个空间,然后再把字符串拷贝给char2,char2就在栈区。

(2)char2是数组名,对地址解引用,*char2仍然在栈区!

(3) 被const的修饰变量不一定在常量区,有可能const修饰的是常变量,存放在栈区!

(4)pchar3是在栈上的指针变量,*pchar3=='‘abcd\0’'是:指向的字符串“abcd\0”存放在常量区

(5)ptr1是在栈上的指针变量,ptr1由于是由malloc出来的,所以ptr1指向的空间(即:*ptr1)在堆区

在这里插入图片描述

二、C/C++使用new和delete的原因

new和delete是C++向内存申请空间和释放空间的操作符, C++为什么要使用new和delete?

1.C语言使用malloc、calloc、realloc、free管理的不便之处在于:

(1)需要手动申请空间,手动计算申请的字节大小。

(2)对返回值void*需要进行强转,否则无法进行解引用。

(3)内存申请是否成功还需要进行判断。

(4)需要include头文件<stdlib.h>

2.new和delete对自定义类型也会进行处理:

(1)new会调用构造函数对类对象进行初始化

(2)delete会调用析构函数进行资源清理

三、C++动态管理内存的方式

1.new和delete

(1)new/delete和malloc/free对内置类型的操作没有区别
注意:
①申请和释放单个空间,需要用new和delete。
②申请和释放多个空间,需要用new [ ]和delete [ ]

(2)new的特点:

#define _CRT_SECURE_NO_WARNINGS 1 
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
using namespace std;


int main()
{
	// 1.C语言申请空间
	int* p0 = (int*)malloc(sizeof(int) * 4);
	if (p0 == NULL)
	{
		perror("malloc fail");
		return 1;
	}

	free(p0);

	// 2.new 1个int整型的元素
	int* p1 = new int;

	delete p1;

	// 3.new 10个int整型元素,(即:数组)
	int* p2 = new int[10];

	// 申请的时候为:new[ ] ,释放的时候,必须要用delete[ ] 
	delete[] p2;

	// 4.可以控制初始化
	int* p3 = new int(10); // new一个int整型 并且初始化为10
	delete p3;
	// 5.new失败后会抛异常,不必手动检查
}

C++11开始支持new初始化数组:
如果没有初始化到的,默认为0

int* p4 = new int[10] {1, 2, 3, 4, 5};
delete[] p4;

在这里插入图片描述
(3)new/delete和malloc/free对自定义类型的操作有区别:

①malloc/free对自定义类型的操作只会开空间/释放空间

②new操作自定义类型会开空间+构造函数初始化,delete操作自定义类型会调用析构函数清理+释放空间

class ListNode
{
public:
	ListNode* _prev;
	ListNode* _next;
	int _val ;

	ListNode()
		:_prev(nullptr)
		,_next(nullptr)
		,_val(0)
	{
		cout << "this _val=" << _val << endl;
	}
	~ListNode()
	{
		cout << "~ListNode()" << endl;
	}
};

int main()
{
	ListNode* p1 = (ListNode*)malloc(sizeof(ListNode) * 4);
	ListNode* p2 = new ListNode;

	free(p1);
	delete(p2);

	return 0;
}

在这里插入图片描述
当程序走完第94行之后,那么p2里面的值都被初始化了。然而对于93行malloc出来的值都是随机值。从调试的界面可以看出,它也调用了构造函数,所以打印出:this _val=0

这就是new和malloc的区别,new对自定义类型会调用构造函数(delete会调用析构函数),对对象进行初始化和空间清理,而C语言之前的malloc和free不会这样做,他们只会开辟空间和释放空间。

2.operator new和operator delete

(1)定义operator new 和operator delete是系统提供的全局函数,new在底层调用operator new全局函数来申请空间,(operator new是对malloc的封装),delete在底层通过operator delete全局函数来释放空间,(operator delete是对free的封装)它们不是new和delete的重载。
(2)与malloc区别:operator new、operator delete的用法和malloc、free的用法是一样的,功能都是在堆上申请释放空间,但是失败了的处理方式不一样,malloc失败返回NULL,operator new失败以后抛异常。
使用new为自定义类型ListNode申请空间并初始化:new会调用operator new函数和构造函数,operator new会调用malloc和失败抛异常机制,因此new和operator new申请失败都会抛异常:
在这里插入图片描述
从上图可以看出,new并不直接调用 malloc,而是调用了operator new,因为operator new被封装了,malloc申请失败直接报返回值,但operator new申请失败会抛异常

在这里插入图片描述
总结:
对于自定义类型来说:(只有自定义类型才有构造和析构,内置类型不存在的)
(1)new—>operator new<—>malloc+构造函数
(2)delete—>析构函数+operator delete<—>free 注意两者的顺序不同

3.总结

malloc/free和new/delete总结:
(1)malloc/free是函数,new/delete是操作符

(2)malloc申请的空间不会初始化,new可以初始化

(3)malloc申请的空间需要手动计算并传递过去,而new只需要在后面跟上空间类型即可。

(4)new不需要强转,而malloc需要

(5)malloc申请空间之后需要判断是否申请失败,new不需要,但是new需要捕获异常

(6)申请自定义类型对象时,malloc/free只会开辟空间,而new在申请空间 后会调用构造函数完成对象的初始化,delete在释放空间 前 会调用析构函数完成空间中资源的清理。

五、内存泄漏

1.定义:内存泄漏的概念:一块已经不用的空间,没有释放

2.危害:长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会 导致响应越来越慢,最终卡死。

3.有关内存泄漏的例题

class A
{
public:
	A()
	{
		cout << "A()" << endl;
	}
	~A()
	{
		cout << "~A()" << endl;
	}
};

int main()
{
	A* a = new A[5];

	//delete a;  内存泄漏,只释放了1次
	delete[] a;

	return 0;
}

关键点:(记住:一定要匹配使用,如若违背结果将不确定!)

new—>delete
new[ ] —>delete[ ]
在这里插入图片描述


好了,今天的分享就到这里了
如果对你有帮助,记得点赞👍+关注哦!
我的主页还有其他文章,欢迎学习指点。关注我,让我们一起学习,一起成长吧!
在这里插入图片描述

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

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

相关文章

帧同步原理

帧同步和状态同步区别 状态同步&#xff1a;发操作&#xff0c;收状态 帧同步&#xff1a;发操作&#xff0c;收操作 逻辑严格排序 经常会有需要排序的列表或者数组&#xff0c;比如攻击距离自己最近的敌人&#xff0c;这时候就需要将身边的敌人进行距离排序&#xff0c;一般…

在having、select子句中使用子查询

目录 在having子句中使用子查询 统计出部门平均工资高于公司平均工资的部门编号、平均工资、部门人数 在select子句中使用子查询 查询每个员工的编号、姓名、职位、部门名称 Oracle从入门到总裁:https://blog.csdn.net/weixin_67859959/article/details/135209645 在havin…

2024.2.26

1、实现信号灯集 sem.c #include<myhead.h>union semun {int val; /* Value for SETVAL */struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */unsigned short *array; /* Array for GETALL, SETALL */struct seminfo *__buf; /* Buffer …

备考2025年考研数学(一)真题练习和解析——填空题

今天距离2025年考研预计还有10个月的时间&#xff0c;看起来挺长&#xff0c;但是对于备考2025年考研的同学来说&#xff0c;必须用好每一天。为了帮助大家提升考研数学一的成绩&#xff0c;我收集整理了1987-2024年的考研数学一的真题和解析&#xff0c;并把2015-2024年十年的…

一个39岁程序员的自白,大龄程序员的出路在哪里?

一个39岁程序员的自白&#xff0c;大龄程序员的出路在哪里&#xff1f; 大龄程序员&#xff0c;最悲惨的&#xff0c;可能是但凡你发个贴&#xff0c;下面就会有类似这种人来怼你 本文来自知乎一个大龄程序员老哥&#xff08;白圣君&#xff09;的自白&#xff0c;涤生哥已经经…

现在骨传导耳机什么牌子最好?专业选购指南与避坑策略

近些年来&#xff0c;耳机行业飞速发展&#xff0c;耳机已经成为日常生活中不可或缺的伙伴&#xff0c;无论是休闲时刻还是运动健身&#xff0c;耳机都伴随着我们。然而长时间使用传统的入耳式耳机可能会导致听力受损、容易脱落和不卫生等问题。为了解决这些问题&#xff0c;满…

【GameFramework框架内置模块】3、数据表(Data Table)

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【GameFramework框架】系列教程目录&#xff1a; https://blog.csdn.net/q7…

3 easy 26. 删除有序数组中的重复项

双指针&#xff1a; //给你一个 非严格递增排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。元素的 相对顺序 应该保持 //一致 。然后返回 nums 中唯一元素的个数。 // // 考虑 nums 的唯…

服务器防漏扫

什么是漏扫&#xff1f; 漏扫是漏洞扫描的简称。漏洞扫描是一种安全测试方法&#xff0c;用于发现计算机系统、网络或应用程序中的潜在漏洞和安全弱点。通过使用自动化工具或软件&#xff0c;漏洞扫描可以检测系统中存在的已知漏洞&#xff0c;并提供相关的报告和建议&#xf…

如何在Linux部署Portainer并结合内网穿透远程管理本地Docker容器

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 Portainer 是一个轻量级的容器管理工具&#xff0c;可以通过 Web 界面对 Docker 容器进行管理和监控。它提供了可…

【牛牛送书 | 第四期】《高效使用Redis:一书学透数据存储与高可用集群》带你快速学习使用Redis

前言&#xff1a; 当今互联网技术日新月异&#xff0c;随着数据量的爆炸式增长&#xff0c;如何高效地存储和管理数据成为了每个公司都必须面对的挑战。与此同时&#xff0c;用户对于应用程序的响应速度和稳定性要求也越来越高。在这个背景下&#xff0c;Redis 作为一个…

C++ 学习之函数对象

C 函数对象基本概念 在C中&#xff0c;函数对象&#xff08;Function Objects&#xff09;是一种类或结构体&#xff0c;它重载了函数调用运算符operator()&#xff0c;因此可以像函数一样被调用。函数对象有时也被称为仿函数&#xff08;Functor&#xff09;。 以下是关于C函…

备战蓝桥杯---基础算法刷题2

题目有一点水&#xff0c;不过还是有几个好题的&#xff0c;我在这分享一下&#xff1a; 很容易想到先往最高处跳再往最低处跳&#xff0c;依次类推&#xff0c;那怎么保证其正确性呢&#xff1f; 证法1. 在此&#xff0c;我们从0开始&#xff0c;假设可以跳到a,b,c(a<b<…

Linux中消息队列

消息队列 概念 消息队列是System V IPC对象的一种消息队列有消息队列ID来唯一标识消息队列就是一个消息列表。用户可以在消息队列中添加消息、读取信息等消息队列可以按照类型来发送/结束消息 消息队列使用步骤 打开/创建消息队列 msgget向消息队列发送信息 msgsnd从消息队…

4.4 MySQL存储

目录 1、使用前提 2、使用连接数据库最初步骤 2.1 最初步骤 2.2 connect()方法中参数简单传递 3、创建数据库(创建架构)和创建表 3.1 创建数据库(创建架构) 3.2 创建表 3.2.1 基本创建 3.2.2 创建自增主键 4、Pycharm 可视化连接 MySQL 图形界面 5、插入、更新、查询…

项目管理工具git

git 1. git介绍1.1. 版本控制系统 2. 创建本地版本库2.1 概念2.2 操作步骤 3. 修改文件4. 练习: 添加一个本地项目到仓库5. 添加远程仓库5.1 添加远程仓库5.2 本地仓库同步到远程仓库5.3 克隆远程仓库到本地5.4 SSH设置 6. 分支管理6.1 创建分支6.2 切换分支6.3 合并分支6.4 解…

基于yolov5的苹果检测(pytorch框架)【python源码+UI界面+功能源码详解】

功能演示&#xff1a; 基于yolov5的苹果检测系统&#xff0c;系统既能够实现图像检测&#xff0c;也可以进行视屏和摄像实时检测_哔哩哔哩_bilibili &#xff08;一&#xff09;简介 基于yolov5的苹果检测系统是在pytorch框架下实现的&#xff0c;这是一个完整的项目&#xf…

Go语言必知必会100问题-06 生产者端接口

生产者端接口 Go语言必知必会100问题-05 接口污染中介绍了程序中使用接口是有价值的。在编码的时候&#xff0c;接口应该放在哪里呢&#xff1f;这是Go开发人员经常有误解的一个问题&#xff0c;本文将深入分析该问题。 在深入探讨问题之前&#xff0c;先对提及的术语做一个定…

如何使用Inno Setup制作Unity构建程序的Windows安装程序

1. 准备 &#xff08;1&#xff09;准备好Unity构建的程序集合 必须包括&#xff1a; Data文件夹&#xff08;xxx_Data&#xff09; Mono文件夹&#xff08;MonoBleedingEdge&#xff09; 打包的应用程序文件&#xff08;xxx.exe&#xff09; Unity播放器dll文件&#xff…

mac flutter 配置

下载Flutter Sdk 直接访问官网无法下载&#xff0c;需要访问中国镜像下载 Flutter SDK 归档列表 - Flutter 中文文档 - Flutter 中文开发者网站 - Flutter Start building Flutter Android apps on macOS - Flutter 中文文档 - Flutter 中文开发者网站 - Flutter 下载后解压…