- 1、本文档共10页,可阅读全部内容。
- 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
深入C基础_new运算符
深入C++的newnew是C++的一个关键字,同时也是操作符关于new的话题非常多,因为它确实比较复杂,也非常神秘,下面我将把我了解到的与new有关的内容做一个总结new的过程当我们使用关键字new在堆上动态创建一个对象时,它实际上做了三件事:获得一块内存空间调用构造函数返回正确的指针当然,如果我们创建的是简单类型的变量,那么第二步会被省略假如我们定义了如下一个类A:class A{ int i;public: A(int _i) :i(_i*_i) {} void Say() { printf(i=%dn, i); }};//调用new:A* pa = new A(3);那么上述动态创建一个对象的过程大致相当于以下三句话(只是大致上):虽然从效果上看,这三句话也得到了一个有效的指向堆上的A对象的指针pa,但区别在于,当malloc失败时,它不会调用分配内存失败处理程序new_handler,而使用new的话会的因此我们还是要尽可能的使用new,除非有一些特殊的需求new的三种形态到目前为止,本文所提到的new都是指的new operator或称为new expression,但事实上在C++中一提到new,至少可能代表以下三种含义:new operatoroperator newplacement newnew operator就是我们平时所使用的new,其行为就是前面所说的三个步骤,我们不能更改它但具体到某一步骤中的行为,如果它不满足我们的具体要求时,我们是有可能更改它的三个步骤中最后一步只是简单的做一个指针的类型转换,没什么可说的,并且在编译出的代码中也并不需要这种转换,只是人为的认识罢了但前两步就有些内容了new operator的第一步分配内存实际上是通过调用operator new来完成的,这里的new实际上是像加减乘除一样的操作符,因此也是可以重载的operator new默认情况下首先调用分配内存的代码,尝试得到一段堆上的空间,如果成功就返回,如果失败,则转而去调用一个new_hander,然后继续重复前面过程如果我们对这个过程不满意,就可以重载operator new,来设置我们希望的行为例如:class A{public: void* operator new(size_t size) { printf(operator new calledn); return ::operator new(size); }};A* a = new A();这里通过::operator new调用了原有的全局的new,实现了在分配内存之前输出一句话全局的operator new也是可以重载的,但这样一来就不能再递归的使用new来分配内存,而只能使用malloc了:void* operator new(size_t size){ printf(global newn); return malloc(size);}相应的,delete也有delete operator和operator delete之分,后者也是可以重载的并且,如果重载了operator new,就应该也相应的重载operator delete,这是良好的编程习惯new的第三种形态placement new是用来实现定位构造的,因此可以实现new operator三步操作中的第二步,也就是在取得了一块可以容纳指定类型对象的内存后,在这块内存上构造一个对象,这有点类似于前面代码中的p-A::A(3);这句话,但这并不是一个标准的写法,正确的写法是使用placement new:#include new.hvoid main(){ char s[sizeof(A)]; A* p = (A*)s; new(p) A(3); //p-A::A(3); p-Say();}对头文件new或new.h的引用是必须的,这样才可以使用placement new这里new(p) A(3)这种奇怪的写法便是placement new了,它实现了在指定内存地址上用指定类型的构造函数来构造一个对象的功能,后面A(3)就是对构造函数的显式调用这里不难发现,这块指定的地址既可以是栈,又可以是堆,placement对此不加区分但是,除非特别必要,不要直接使用placement new ,这毕竟不是用来构造对象的正式写法,只不过是new operator的一个步骤而已使用new operator地编译器会自动生成对placement new的调用的代码,因此也会相应的生成使用delete时调用析构函数的代码如果是像上面那样在栈上使用了placement new,则必须手工调用析构函数,这也是显式调用析构函数的唯一情况:p
文档评论(0)