发现一个new的用法,如下
int val;
new ((void*)&val) int;
搜索后知道它叫 定位new运算符,作用就是在指定地址new一块内存。
废话不多说,看看测试代码:
测试代码1
int main() {
int a;
int* pa = new ((void*)&a) int;
std::cout << "&a: " << (void*)&a << std::endl;
std::cout << "pa: " << pa << std::endl;
delete pa; //这里会运行报错
}
运行结果如下:
可见:
- a的地址和pa是一样的,都是栈区地址。
- pa以a的地址为开始开辟了一个int大小的内存。
- pa指向栈区,所以delete时报错:invalid pointer; 所以,这种在非空闲内存上定位new的内存,不需要手动释放,它的生存周期和a一样。
测试代码2
问:在堆区定位new一个内存能否delete? 看测试代码:
#include <iostream>
int main() {
int* pb = new int;
int* pb2 = new((void*)pb) int;
std::cout << "pb: " << pb << std::endl;
std::cout << "pb2: " << pb2 << std::endl;
delete pb2;
delete pb;
}
运行结果如下:
可见:
- 指向堆区的定位new指针是可以正常释放的(因为重复释放了)。
测试代码3
问: 这种new会运行构造函数吗?
#include <iostream>
class Dog
{
public:
Dog(){
std::cout << "dog burn\n";
}
};
int main() {
char buf[sizeof(Dog)];
Dog* pDog = new (&buf) Dog;
std::cout << &buf << std::endl;
std::cout << pDog << std::endl;
}
运行结果:
可见:
- 会执行构造函数
其他小问题自己验证吧。
总结:
- 在指定的地址new一个内存,不检测该内存是否被占用
- 一般使用情况下不需要delete
- 会执行构造函数