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

本文探讨了C++中的定位new运算符,它允许在指定地址分配内存。通过测试代码展示了栈区和堆区的定位new用法,以及其内存释放规则。定位new不会检查目标内存是否已占用,且在某些情况下不需要手动删除。此外,它会调用构造函数初始化对象。总结了定位new的关键特性,包括内存分配、构造函数执行以及内存管理的特殊性。
摘要由CSDN通过智能技术生成

发现一个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
  • 会执行构造函数
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值