【C++】——内存管理

目录

    • 回忆C语言内存管理
    • C++内存管理方式new delete
    • operator new与operator delete函数
    • new和delete的实现原理
    • 定位new表达式(placement-new)
    • malloc/free和new/delete的区别

回忆C语言内存管理

void Test()
{int* p1 = (int*)malloc(sizeof(int));free(p1);int* p2 = (int*)calloc(4, sizeof(int));//不用freeint* p3 = (int*)realloc(p2, sizeof(int) * 10);free(p3);
}

在这里插入图片描述

C++内存管理方式new delete

通过new和delete操作符进行动态内存管理

C++自动计算大小,不需要进行强制类型转换

int main()
{//动态申请一个int型的空间int* p1 = (int*)malloc(sizeof(int));//Cint* p2 = new int;//C++//动态申请10个int型空间int* p3 = (int*)malloc(sizeof(int) * 10);//Cint* p4 = new int[10];//C++free(p1);free(p3);delete p2;delete[] p4;return 0;
}

C++额外支持开空间和初始化

int main()
{int* p1 = new int(10);int* p2 = new int[10] {1, 2, 3};int* p3 = new int[10] {};return 0;
}

malloc没有办法很好的支持动态申请的自定义对象初始化
自定义类型,开空间+调用构造函数初始化
自定义类型,调用析构函数+释放空间

class A
{
public:A(int a = 1){cout << "A(int a = 1)" << endl;}~A(){cout << "~A()" << endl;}
private:int _a;
};
int main()
{//自定义类型,开空间+调用构造函数初始化A* p1 = new A;A* p2 = new A(3);//自定义类型,调用析构函数+释放空间delete p1;delete p2;return 0;
}
int main()
{A aa1(1);A aa2(2);A* p1 = new A[10]{ aa1,aa2 };delete[] p1;A* p2 = new A[10]{ A(1),A(2) };delete[] p2;A* p3 = new A[10]{ 1,2 };delete[] p3;return 0;
}

栈:

typedef int DataType;
class Stack
{
public:Stack(size_t capacity=4){cout << "Stack(size_t capacity)" << endl;_array = new DataType[capacity];_size = 0;_capacity = capacity;}void Push(int x){_array[_size++] = x;}~Stack(){cout << "~Stack()" << endl;delete[] _array;_size = _capacity = 0;}
private:DataType* _array;size_t _size;size_t _capacity;
};
Stack* func()
{int n = 0;cin >> n;Stack* pst = new Stack(n);return pst;
}
int main()
{Stack* ptr = func();ptr->Push(1);ptr->Push(2);delete ptr;return 0;
}

在这里插入图片描述

operator new与operator delete函数

new和delete是用户进行动态内存申请和释放的操作符,operator new 和operator delete是系统提供的全局函数,new在底层调用operator new全局函数来申请空间,delete在底层通过operator delete全局函数来释放空间。

typedef int DataType;
class Stack
{
public:Stack(size_t capacity=4){cout << "Stack(size_t capacity)" << endl;_array = new DataType[capacity];_size = 0;_capacity = capacity;}void Push(int x){_array[_size++] = x;}~Stack(){cout << "~Stack()" << endl;delete[] _array;_size = _capacity = 0;}
private:DataType* _array;size_t _size;size_t _capacity;
};
Stack* func()
{int n = 0;cin >> n;Stack* pst = new Stack(n);return pst;
}
int main()
{Stack* pst1 = (Stack*)operator new(sizeof(Stack));operator delete(pst1);Stack* pst2 = new Stack;delete pst2;return 0;
}

operator new 实际也是通过malloc来申请空间,如果malloc申请空间成功就直接返回,否则执行用户提供的空间不足应对措施,如果用户提供该措施 就继续申请,否则就抛异常。
operator delete 最终是通过free来释放空间的。

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

new和delete形式要匹配使用

new和delete的实现原理

1、内置类型:

如果申请的是内置类型的空间,new和malloc,delete和free基本类似。不同的地方是:new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申请空间失败时会抛异常,malloc会返回NULL

2、自定义类型:

new的原理:
1、 调用operator new函数申请空间
2、在申请的空间上执行构造函数,完成对象的构造
delete的原理:
1、在空间上执行析构函数,完成对象中资源的清理工作
2、调用operator delete函数释放对象的空间
new T[N]的原理:
1、调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请
2、在申请的空间上执行N次构造函数
delete[]的原理:
1、在释放的对象空间上执行N次析构函数,完成N个对象中资源的清理
2、调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间

定位new表达式(placement-new)

定位new表达式是在已分配的原始内存空间中调用构造函数初始化一个对象。
使用格式:new (place_address) type或者new (place_address) type(initializer-list)
place_address必须是一个指针,initializer-list是类型的初始化列表
**使用场景:**定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如
果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化

typedef int DataType;
class Stack
{
public:Stack(size_t capacity=4){cout << "Stack(size_t capacity)" << endl;_array = new DataType[capacity];_size = 0;_capacity = capacity;}void Push(int x){_array[_size++] = x;}~Stack(){cout << "~Stack()" << endl;delete[] _array;_size = _capacity = 0;}
private:DataType* _array;size_t _size;size_t _capacity;
};
Stack* func()
{int n = 0;cin >> n;Stack* pst = new Stack(n);return pst;
}int main()
{Stack* pst1 = (Stack*)operator new(sizeof(Stack));//pst1->Stack(4);//不支持new(pst1)Stack(4);//显示调用构造函数pst1->~Stack();operator delete(pst1);return 0;
}

malloc/free和new/delete的区别

共同点:
malloc/free和new/delete 都是从堆上申请空间,并且需要手动释放
不同点:
1、malloc和free是函数,new和delete是操作符
2、malloc申请的空间不会初始化,new可以初始化
3、malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可,如果是多个对象,[]中指定对象个数即可
4、malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
5、malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new需要捕获异常
6、申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间中资源的清理

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

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

相关文章

攻防世界-reverse-logmein

题目描述&#xff1a;菜鸡开始接触一些基本的算法逆向了 下载附件&#xff0c;是一个可执行程序 1. 思路分析 逆向出来看看代码 从代码中来看&#xff0c;密码长度需要和V8相等&#xff0c;并且每一个字符的运算结果需要满足 s[i] (char)(v8[i % v6 - 8] ^ v8[i]) 但是这…

openGauss学习笔记-31 openGauss 高级数据管理-索引

文章目录 openGauss学习笔记-31 openGauss 高级数据管理-索引31.1 语法格式31.2 参数说明31.3 示例 openGauss学习笔记-31 openGauss 高级数据管理-索引 索引是一个指向表中数据的指针。一个数据库中的索引与一本书的索引目录是非常相似的。 索引可以用来提高数据库查询性能&…

从零开始理解Linux中断架构(24)软中断核心函数__do_softirq

1)概要 __do_softirq函数处理是总是尽可能的执行所有未决软中断。 (1)关闭软中断:在preempt_count设置软中断标志:SOFTIRQ_OFFSET 让in_interrupt检查条件为真,进入软中断处理临界区,后面进来的处理请求,需要检查in_interrupt(),从而达到禁止本cpu上的软中断嵌套的目…

【雕爷学编程】MicroPython动手做(33)——物联网之天气预报

天气&#xff08;自然现象&#xff09; 是指某一个地区距离地表较近的大气层在短时间内的具体状态。而天气现象则是指发生在大气中的各种自然现象&#xff0c;即某瞬时内大气中各种气象要素&#xff08;如气温、气压、湿度、风、云、雾、雨、闪、雪、霜、雷、雹、霾等&#xff…

Java三大特征之继承【超详细】

文章目录 一、继承概念二、继承的语法三、父类成员访问3.1子类中访问父类的成员变量3.2子类和父类成员变量同名3.3子类中访问父类的成员方法 四、super关键字五、子类构造方法六、super和this七、再谈初始化八、protected 关键字九、继承方式十、final 关键字十一、继承与组合 …

springboot+vue农业技术信息管理系统_9927h

随着信息时代的发展&#xff0c;计算机迅速普及&#xff0c;传统的农业信息管理方式显得不够快捷&#xff0c;这时我们就需要创造更加便利的管理方法&#xff0c;对农业信息进行统计&#xff0c;便于统一管理。将传统管理方式转变为信息、智能化显得尤为重要&#xff0c;农业信…

选读SQL经典实例笔记18_Exactly

1. 问题9 1.1. 只讲授一门课程的教授 1.2. sql select p.*from professor p,teach twhere p.lname t.lnameand p.lname not in ( select t1.lnamefrom teach t1,teach t2where t1.lname t2.lnameand t1.cno &#xff1e; t2.cno ) LNAME DEPT SALARY …

MIT 6.824 -- MapReduce -- 01

MIT 6.824 -- MapReduce -- 01 引言抽象和实现可扩展性可用性(容错性)一致性MapReduceMap函数和Reduce函数疑问 课程b站视频地址: MIT 6.824 Distributed Systems Spring 2020 分布式系统 推荐伴读读物: 极客时间 – 大数据经典论文解读DDIA – 数据密集型应用大数据相关论文…

LeetCode 热题 100 JavaScript--102. 二叉树的层序遍历

给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历 。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[[3],[9,20],[15,7]] 示例 2&#xff1a; 输入&#xff1a;root [1…

C#核心知识回顾——19.插入排序

1.插入排序的基本原理 871542639 两个区域 排序区 未排序区 用一个索引值做分水岭 未排序区元素 与排序区元素比较 插入到合适位置 直到未排序区清空 int[] arr { 8, 6, 7, 2, 9, 4 };//第一步//能取出未排序区…

Unity面试题:热更新篇

请简要介绍Unity热更新的原理和实现方式。 答&#xff1a;Unity热更新的原理是通过将游戏的资源和代码分离&#xff0c;将代码部分放置在服务器端&#xff0c;游戏启动时通过网络下载更新的代码并动态加载&#xff0c;以达到实现热更新的目的。实现方式包括AssetBundle、ILRunt…

CVE漏洞复现-CVE-2019-5021 镜像漏洞利用

CVE-2019-5021 镜像漏洞利用 随着容器技术的普及&#xff0c;容器镜像也成为软件供应链中非常重要的一个组成的部分。人们像使用 pip 等工具从仓库获取各种编程软件库一样&#xff0c;可以从 Docker Hub 或 第三方仓库拉取镜像&#xff0c;在其基础上进行开发&#xff0c;从而…