C++ 定位new运算符: new (void*) xxx

发现一个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
  • 会执行构造函数

猜你喜欢

转载自blog.csdn.net/qq_40541268/article/details/126149108
今日推荐