C++内存管理(上)

简介: C++内存管理

一、C/C++内存分布


1669269764774.jpg

C语言阶段我们学习过C程序内存区域是如何划分的,C++也是这样划分程序的,只不过在使用方式上和C有所不同,我们先复习C,先看几道题目:

1669269780369.jpg

解析图:


1669269792833.jpg


上面的很多相信大家都可以轻松解决,这里只说几道容易出错的题目。


1、*char2存储在什么地方?


很多老铁可能会这样觉得,char2是常量字符串"abcd"的地址,存储在栈上,然后*char2拿到字符串的首字符'a',所以应该存储在代码段(常量区),其实不然。这里我想问老铁一个问题,如果*char2拿到存储在常量区的'a',那么我如果对char2的内容做修改,是否也要修改常量区的内容呢?显然是不可以的,常量区内容不能被修改。所以*char2应该在栈上!!我们只是把常量字符串"abcd"拷贝到了char2上,*char2还在栈上可以被修改。


2、与之相对应的就是*pchar3,pchar3是一个地址,虽然用const修饰,但他就是一个地址,没有实际内容,指向存储在代码段的常量字符串。因为他用const修饰,限制修改,没必要再拷贝。


【说明】:


1、栈又叫堆栈--非静态局部变量/函数参数/返回值等等,栈是向下增长的。


2、内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享内存,做进程间通信。


3、堆用于程序运行时动态内存分配,堆是向上增长的。


4、数据段--存储全局数据和静态数据。


5、代码段--可执行的代码/只读常量。


二、C++内存管理方式


C++兼容C语言,C语言使用malloc、calloc、realloc管理内存,C++中可以继续使用这些内存管理方式,但是有些地方用它就不太方便,于是C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理。


Ⅰ new和delete操作内置类型


C++提出用new来开辟空间,用delete来释放空间。那么我们怎么使用new和delete呢?


①new和delete的基本使用

new默认不会初始化。

1669269828326.jpg

new给初始化的方式是在内置类型后面加(),()里面是要初始化的值。

1669269841361.jpg

new想要开辟多个空间,需要加[ ],同样在delete时也需要加[ ]。


new在初始化一个空间时用(),初始化多个空间时,类似于数组,用{ },里面放初始化值,没有给初始化的后续默认是给0,如果压根没有给初始化值,那么开辟的多个空间里面默认随机值。

1669269853932.jpg


Ⅱ new和delete操作自定义类型


我们光看new和delete对内置类型,与C语言那一套难分高下,可能有优势的地方在于可以自行给初始值,而calloc最多都设为0。new和delete主要是针对自定义类型的。


1669269882652.jpg


我们看到new/delete 和malloc/free最大区别是 new/delete对于自定义类型除了开空间,还会调用构造和析构函数,这一点会给我们带来很大方便。


举个例子:之前在C语言数据结构阶段构造链表时,我们每开辟一个新节点,就要调用函数对它初始化,这里在C++我们开辟节点,就可以自动调用构造初始化,很方便。

struct ListNode
{
  ListNode(int val)
  :_next(nullptr)
  , _val(val)
  {
  }
  ListNode* _next;
  int _val;
};
int main()
{
  ListNode* n1 = new ListNode(1);
  ListNode* n2 = new ListNode(2);
  ListNode* n3 = new ListNode(3);
  ListNode* n4 = new ListNode(4);
  n1->_next = n2;
  //...
  //不用写buynode
  return 0;
}


Ⅲ new和delete要匹配使用


new和delete配套使用,用于开辟一块和销毁一块空间。new[ ]和delete[ ]配套使用,用于开辟多块和销毁多块空间。


new[ ]和delete

1669269910834.jpg

这里就会报错,析构失败。


但是,这里有个奇怪的现象,我把析构函数屏蔽掉就不会报错!

1669269921558.jpg

这里是vs编译器默认的析构函数对此处有优化,它和编译器的实现有关,不是很重要,重要的是这里一定要匹配使用,不要随意匹配,否则任何错误都有可能发生。

相关文章
|
24天前
|
存储 缓存 编译器
【硬核】C++11并发:内存模型和原子类型
本文从C++11并发编程中的关键概念——内存模型与原子类型入手,结合详尽的代码示例,抽丝剥茧地介绍了如何实现无锁化并发的性能优化。
|
5天前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
|
2月前
|
存储 缓存 C语言
【c++】动态内存管理
本文介绍了C++中动态内存管理的新方式——`new`和`delete`操作符,详细探讨了它们的使用方法及与C语言中`malloc`/`free`的区别。文章首先回顾了C语言中的动态内存管理,接着通过代码实例展示了`new`和`delete`的基本用法,包括对内置类型和自定义类型的动态内存分配与释放。此外,文章还深入解析了`operator new`和`operator delete`的底层实现,以及定位new表达式的应用,最后总结了`malloc`/`free`与`new`/`delete`的主要差异。
65 3
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
199 4
|
3月前
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
207 22
|
3月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
3月前
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
63 0
【C++打怪之路Lv6】-- 内存管理
|
3月前
|
存储 C语言 C++
【C/C++内存管理】——我与C++的不解之缘(六)
【C/C++内存管理】——我与C++的不解之缘(六)
|
3月前
|
程序员 C语言 C++
C++入门5——C/C++动态内存管理(new与delete)
C++入门5——C/C++动态内存管理(new与delete)
104 1
|
3月前
|
编译器 C语言 C++
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
594 1