zoukankan      html  css  js  c++  java
  • New

    今天,我们只讲new。

    1. new operator

    new操作符(new operator)比较容易理解,相信对C++比较熟悉的人,对下面的代码都不会觉得陌生。这是使用new 最常见的形式。
    class A{
    ....
    };
    std::shared_ptr<A> a(new A);
    new operator实际上就是: 分配内存+调用构造函数初始化。
    2. operator new
    new operator和operator new虽然差不多,但是你不可以把它们等同起来。与new operator相比,operator new 少了构造初始化的部分,它与malloc比较像。
    operator new的声明一般为:
    void * operator new(size_t size);
    需要注意的是,operator new返回一个未经过处理的指针(类型是void*)。new operator调用operator new来分配内存,operator new也能够被重载。
    默认的operator new 重载版本有:
    void * operator new(std::size_t count) throw(std::bad_alloc); //normal new
    void * operator new(std::size_t count, void *ptr)throw(); //placement new
    void * operator new(std::size_t count, const std::nothrow_t&)throw();//nothrow new, 为了兼容以前的版本
    我们来看一下operator new的源码:
    void* operator new(std::size_t) _GLIBCXX_THROW (std::bad_alloc)
    __attribute__((__externally_visible__));
    void* operator new[](std::size_t) _GLIBCXX_THROW (std::bad_alloc)
    __attribute__((__externally_visible__));
    •  _GLIBCXX_THROW: 指定该函数抛出的异常,C++11后该宏为空
    • __attribute__: GCC使用__attribute__关键字来描述函数,变量和数据类型的属性,用于编译器对源代码的优化。
    • __externally_visible__: GCC的优化参数,详情:https://gcc.gnu.org/onlinedocs/gcc-4.7.2/gcc/Function-Attributes.html
     
    3. placement new
    所谓定位放置(placement new)就是在用户指定的内存位置上构建新的对象,这个构建过程不需要额外分配内存,只需要调用对象的构造函数即可。
    class A{
    ....
    };
    int main(){
    char mem[100];
    A* p=new (mem) A;
    p->~A();
    return 0;
    }
    operator new和placement new的关系:
    class A{
    ....
    };
    
    A* p=new A(10);//有参构造
    //等同于下面的调用
    void* mem=malloc(sizeof(A));
    A* p=new (mem) A(10);
    看似神奇的placement new实际上是operator new的一个全局重载版本,它实际上做了这么一件事(我们来看一下源码):
    // Default placement versions of operator new.
    inline void* operator new(std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT
    { return __p; }
    inline void* operator new[](std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT
    { return __p; }

     我们可以看到实际上它不加任何处理直接返回了指针。那么函数的两个参数从哪来?

    其实,当我们写了下面的代码时:
    A* p=new (mem) A;
    实际上,转变为了:
    operator new (sizeof(A), men);
    到这里,就知道上面源码的operator new两个参数是什么了吧。
    4. 重写operator new
    这部分省略(网上的资料有不少):
    https://www.cnblogs.com/cly-blog/p/5984660.html
    5. 总结
    看到这里,你是否对new operator、operator new、placement new有了一个全面的了解,下面我们来总结一下:
    • new operator:分配内存+调用构造函数
    • operator new:分配内存
    • placement new:在已分配的内存里调用构造函数
    • 假设你想定制自己的在堆对象被建立时的内存分配过程,你应该写你自己的operator new函数。然后使用new操作符,new操作符会调用你定制的operator new。

     
    生命中所有的灿烂,终要寂寞偿还。——《百年孤独》
  • 相关阅读:
    【Leetcode】Unique Binary Search Trees
    linux C函数之access函数的用法
    Dispatcher.BeginInvoke()方法使用不当导致UI界面卡死的原因分析
    【Leetcod】Unique Binary Search Trees II
    KVM客户机使用主机USB设备
    运行Maven是报错:No goals have been specified for this build
    SQL2008R2 express版本不支持维护计划
    已超过了锁请求超时时段的原因
    Oracle免客户端InstantClient安装使用
    将存储过程的返回值赋给变量
  • 原文地址:https://www.cnblogs.com/ZWJCNBLOG/p/11509007.html
Copyright © 2011-2022 走看看