C++基础 之 C++ 中的内存管理问题

简介: C++基础 之 C++ 中的内存管理问题

C和C++都有的内存划分:

b80446426a294615be742d025f32aa57.png

不同的数据存在不同的区域更方便管理和使用(这个图也叫做进程虚拟地址空间)

1. char* ch = "abcd";
2. char arr[] = "abcd";


大家觉得这个 ch *ch arr *arr 他们会是在哪个位置上的呢?

ch 是在  *ch 是在常量区  arr 是在  *arr  是在

ch是一个指针,肯定是在栈上面的,它指向的是一个字符串,字符串是在常量区的

arr是一个数组名,也是在栈上面的,它指向的是一个数组,数组里面的内容是从常量区拷贝过来的,数组本身是在 栈 的

其实我自己也可以通过取地址来判断是不是在一起的


C++中的内存管理:

在c语言的时候我们就学过三个动态内存管理的函数 malloc calloc realloc ,C语言进阶学习日志:动态内存分配_luck++的博客-CSDN博客_c语言内存日志

大家如果有遗忘的可以看看。

New和Delete:

c++开辟空间的方法是使用的是new:

int* a = new int;//new一个int对象
int* a = new int[10];//new十个int对象
int* a = new(10);//new一个int对象,初始化为10
int* a = new int[10]{10,1,2,3};//new十个int对象,初始化为10,1,2,3,0,0,0,0,0,0

大家看到这里有的是一次性开辟多个对象,有的是一次开一个,对于这种情况,delete在着的用法要注意:对于一次性开多个对象的,应该用delete[],一次开一个的就用delete就可以,注意这两者是一定要匹配的,所说不匹配可能不会报错,但还是建议匹配使用!


其实对于内置类型而言,用malloc和new没有什么区别,他们的区别在于自定义类型 


对于自定义类型:我们的malloc开辟的空间,不会初始化,除非自己显示去写,但new不一样,它不仅会给你开空间还会调用自定义类型的构造函数来初始化(这样就保证了我们的数据每次都能得到初始化)其实大家都看到拷贝构造、构造、析构这些都是为了补c语言的坑

297cf00415e64243a7aafcb9e849697b.png

大家要注意我们的构造函数是不能显示调用的,我们如果是malloc这个自定义类型,往后就不好初始化了,除非你的成员变量是公有的,你也可以用定位new,那这里直接去new是不是便捷很多呢

class date
{
public:
  date(int a = 1)
    :_hour(a)
  {
    _a = new int[10];
  }
private:
  int _hour;
  int* _a;
};
int main()
{
  date* a = new date;
  delete a;
}

 大家来看这块代码 new 和 delete 是怎么执行的:

cde06cf3b03d49069b0247c6e682646b.png

我delete之所以要先去调用析构,就是我们编译器不知道这里的指针是不是指向的开辟出来的空间,所以它要去求助我们自己写的析构函数来对那些特别的变量进行处理,处理完才会释放对象的空间,不这么做就有可能会有内存泄漏的风险,虽说我们程序可以正常运行和结束,但内存泄漏始终是一个问题,你不注意在公司里面发生了内存泄漏就有可能会有象限不到的后果,大家就还是要注意一下 


 其实new还会去调用operator new ;delete 也会去调用 operator delete,注意:这两个可不是重载,他们是new和delete的一种底层机制,后面会说到 


 注意:这里我没有检查 a ,也是因为new出错是不会返回空指针的,它会抛异常,这个时候再去检查是否为空,就没有什么意义了。大家如果想让他不抛异常,可以选择捕获,让他打印错误信息:

try()
{
    //可能会抛异常的地方
}
catch (const exception& e)
{
    cout<<e.what()<<endl;
}


operator new 、operator delete

new和delete是用户进行动态内存申请和释放的操作符,operator new,operator delete它们是系统提供的全局函数


有个点大家要注意我们的operayor new里面封装的是malloc,失败是抛异常

1.     date* d1 = (date*)operator new(sizeof(date));
2.  operator delete(d1);


实际上他们和malloc和free在用法上是没有什么区别的(他们不会去调用构造和析构函数哦)

我们之前不是说过new会先开空间在调用构造嘛,其实意思也就是先调用operator new没问题再调用构造函数,那delete也是一样的道理

class date
{
public:
  date(int a = 1)
  {
    _hour = a;
    _a = new int[10];
  }
private:
  int _hour;
  int* _a;
};

f42c1b2f103f4af6810eb4dececfb059.png

大家看见了吧这里是不是先调用的operator new 再去调用的构造函数哇,我构造函数里面写的是new十个int的空间,这里是不是有个operator new[](这个里面封装的是operator new,调用N次(有多少调多少)构造函数),它的上面的28h,我们把28转换成10进制正好是 40 !!!(operator delete[]也是一样的道理)


总结:它们与malloc和free有这样的区别:申请自定义类型对象时,malloc和free只会开辟空间,不会调用构造和析构函数,new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成空间的清理 


内存池:

现在有一种这样的情况,就是我们反复地去申请空间,又时不时删除前面申请的空间,那我们这样是不是又要反复地区堆里面申请空间


所以有了一个内存池,这个内存池是我们自己设置的(通过改变operator new 来设置),我们每次申请空间都去内存池里面拿,这样相对去堆里面拿,就会更快一些!!!

目录
相关文章
|
2月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
127 26
|
7月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
3月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
60 1
|
9月前
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
401 68
|
6月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
7月前
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
340 0
|
8月前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
328 0
|
10月前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
180 3
|
10月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
548 4
|
11月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。