C++基础 — new的使用


1、地址空间

可执行程序(进程)的虚拟地址空间:

在这里插入图片描述
内核:操作系统
栈区:函数的形参,非静态的局部变量,函数现场保护数据等,栈是向下增长的
共享库的内存映射区域:用于装载一个共享的动态内存库。用户可以使用系统接口创建共享内存,做进程间的通信
堆区:用于程序运行时动态内存的分配,堆是可以上增长的
数据段:存储全局数据和静态数据,分为 .bss 和 .data
代码段:可执行程序(机器指令)和常量数据

2、new

new 负责在 堆区heap 中找到一个足以满足要求的内存。new还有另外一种变体,被称为 定位new 运算符,能够指定使用具体的内存位置。可以用这个特性来设置其内存管理章程,处理需要通过特定地址进行访问的硬件或在特定位置创建对象

2.1 new的基本用法

int *a = new[5];
class A { ... };  // 声明一个类 A
A *obj = new A();  // 使用 new 创建对象
delete []a;  // new 和 delete 必须对应使用,new 创建一组对象,那么 delete 就要释放一组对象
delete obj;

【分析】
这里我们需要注意,new int[5] 仅仅分配了空间,但是 new A(),不仅仅为对象 obj 在堆上分配了空间,而且还调用了 A的构造函数,生成了这个对象

所以 new A() 这样方式的功能如下:

  • 在堆上分配空间
  • 在分配空间上调用对象的构造函数(这也是 new 和 malloc 的主要区别,是否调用构造函数

同理,在调用 delete obj 的时候

  • 首先调用 这个对象 的析构函数
  • 然后释放这个对象的空间

2.2 new的升级版本

从上面可以看到,new的功能是:1. 分配空间; 2. 调用构造函数

其实C++规定new的这两个功能分开来实现:

  1. 分配空间:调用函数 operate new 来实现
  2. 调用构造函数:调用 placement new 来实现

现在有三个 new 了,第一个 new 就是我们常说的 new,这个 new 调用接下来的两个 new 来实现它的功能。(我们称这个 new 为:new operator,叫做 “new 表达式”,因为operator 在 new 后面,所以叫做:new表达式,也就是关键字)

new 关键字会调用 operator new 来分配空间:这里 operator new 是一个全局的函数,写在一个文件中 。当使用 new 关键字的时候,编译器会自动找到这个函数,并且调用这个函数,这个函数的声明如下:

// 全局 operator new
void* operator new(std::size_t size) throw(std::bad_alloc)  {
	if(size == 0)
		size =1;
	void* p;
	while((p = ::malloc(size)) == 0)  {  // 采用 malloc 分配空间
		std::new_handler nh = std::get_new_handler();
		if(nh)
			nh();
		else
			throw std::bad_alloc();
	}
	return p;
}
// 对应的全局 operator delete 采用 free 释放空间
void operator delete(void* ptr)  {
	if (ptr)
		::free(ptr);  // 采用 free 释放空间
}

这个 operator new 函数称为 全局 operator new。(这里称为 全局 主要是因为:每个类还可以重载自己的 operator new() 函数

重载 operator 必须非常的注意: 陈硕的Blog

2.3 内存管理的基本要求

如果只考虑分配和释放,内存管理基本要求是 “不重不漏” :既不重复 delete,也不漏掉 delete。也就是我们常说的 new/delete 要配对,“配对” 不仅是个数相等,还隐含了 new 和 delete 的调用本身要匹配,不要“东家借的西家还”。例如:

  • 用系统默认的 malloc() 分配的内存要交给系统默认的 free() 去释放;
  • 用系统默认的 new 表达式创建的对象要交给系统默认的 delete 表达式去析构并释放;
  • 用系统默认的 new[] 表达式创建的对象要交给系统默认的 delete[] 表达式去析构并释放;
  • 用系统默认的 ::operator new() 分配的内存要交给系统默认的 ::operator delete() 去释放;
  • placement new 创建的对象要用 placement delete 去析构(其实就是直接调用析构函数);
  • 从某个内存池 A 分配的内存要还给这个内存池;

3、对于内置类型 new/delete/malloc/free 可以混用

区别

  • new/delete 是C++中的运算符。malloc/free 是函数
  • malloc 申请内存空间时,手动计算所需大小,new 只需要类型名,自动计算大小
  • malloc 申请的内存空间不会初始化,new 可以初始化(需要调用构造函数)
  • malloc 的返回值为 void*,接受时必须强转,new不需要
  • malloc 申请内存空间失败时,返回的是NULL,使用时必须判空;new申请内存空间失败时会抛出异常(可以加上 nothrow),所以要有捕获异常处理程序

4、C和C++的动态内存管理

4.1 C的动态内存管理

int main()
{
	int n = 10;
	int* ipa = (int*)malloc(sizeof(int)*n);
	int* ipb = (int*)calloc(n, sizeof(int));
	ipa = (int*)realloc(ipa, sizeof(int)*n*2);
	//
	// 
	free(ipa);  ipa = NULL;  // 释放之后一定要将指针置为NULL,防止出现 失效指针
	free(ipb);  ipb = NULL;
	return 0;
}

4.2 C++的动态内存管理

1、new 运算符的使用

int main()
{
	int n = 10;
	int* ipa = new int(10);
}

请添加图片描述

  • 84
    点赞
  • 406
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中的new是一个用于动态申请存储空间的运算符。它有多种用法和注意事项。首先,new可以用于申请一个对象,语法为new+数据类型(初值),返回值为申请空间的对应数据类型的地址。例如,使用new申请一个初值为10的整型数据可以这样写:int *p = new int(10)。\[2\] 此外,new关键字会调用operator new函数来分配空间。operator new是一个全局的函数,当使用new关键字时,编译器会自动找到这个函数并调用它。operator new函数的作用是分配指定大小的内存空间,并返回指向该空间的指针。对应的全局operator delete函数则用于释放通过new分配的内存空间。\[3\] 需要注意的是,使用new关键字申请的内存空间需要手动释放,否则会导致内存泄漏。释放内存空间的方法是使用delete运算符,语法为delete+指针变量。例如,释放前面申请的整型数据的内存空间可以这样写:delete p。\[1\] 总结起来,C++中的new关键字可以用于动态申请存储空间,并返回指向该空间的指针。使用new关键字申请的内存空间需要手动释放,释放的方法是使用delete运算符。\[1\]\[2\]\[3\] #### 引用[.reference_title] - *1* [C++new与delete的使用](https://blog.csdn.net/weixin_56935264/article/details/124858025)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [C++new使用](https://blog.csdn.net/qq_54182926/article/details/119193250)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [C++基础new使用](https://blog.csdn.net/qq_41884147/article/details/125098711)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值