欢迎来到三一办公! | 帮助中心 三一办公31ppt.com(应用文档模板下载平台)
三一办公
全部分类
  • 办公文档>
  • PPT模板>
  • 建筑/施工/环境>
  • 毕业设计>
  • 工程图纸>
  • 教育教学>
  • 素材源码>
  • 生活休闲>
  • 临时分类>
  • ImageVerifierCode 换一换
    首页 三一办公 > 资源分类 > DOCX文档下载  

    C++中new 的用法大全 详细.docx

    • 资源ID:3153658       资源大小:44.81KB        全文页数:14页
    • 资源格式: DOCX        下载积分:6.99金币
    快捷下载 游客一键下载
    会员登录下载
    三方登录下载: 微信开放平台登录 QQ登录  
    下载资源需要6.99金币
    邮箱/手机:
    温馨提示:
    用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)
    支付方式: 支付宝    微信支付   
    验证码:   换一换

    加入VIP免费专享
     
    账号:
    密码:
    验证码:   换一换
      忘记密码?
        
    友情提示
    2、PDF文件下载后,可能会被浏览器默认打开,此种情况可以点击浏览器菜单,保存网页到桌面,就可以正常下载了。
    3、本站不支持迅雷下载,请使用电脑自带的IE浏览器,或者360浏览器、谷歌浏览器下载即可。
    4、本站资源下载后的文档和图纸-无水印,预览文档经过压缩,下载后原文更清晰。
    5、试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。

    C++中new 的用法大全 详细.docx

    C+中new 的用法大全 详细深入C+的new new是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 = (A*)malloc(sizeof(A); pa->A:A(3); return pa; 虽然从效果上看,这三句话也得到了一个有效的指向堆上的A对象的指针pa,但区别在于,当malloc失败时,它不会调用分配内存失败处理程序new_handler,而使用new的话会的。因此我们还是要尽可能的使用new,除非有一些特殊的需求。 new的三种形态 到目前为止,本文所提到的new都是指的new operator或称为new expression,但事实上在C+中一提到new,至少可能代表以下三种含义:new operator、operator new、placement new。 new 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.h> void main char ssizeof(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->A; 当我们觉得默认的new operator对内存的管理不能满足我们的需要,而希望自己手工的管理内存时,placement new就有用了。STL中的allocator就使用了这种方式,借助placement new来实现更灵活有效的内存管理。 处理内存分配异常 正如前面所说,operator new的默认行为是请求分配内存,如果成功则返回此内存地址,如果失败则调用一个new_handler,然后再重复此过程。于是,想要从operator new的执行过程中返回,则必然需要满足下列条件之一: l 分配内存成功 l new_handler中抛出bad_alloc异常 l new_handler中调用exit或类似的函数,使程序结束 于是,我们可以假设默认情况下operator new的行为是这样的: void* operator new(size_t size) void* p = null while(!(p = malloc(size) if(null = new_handler) throw bad_alloc; try new_handler; catch(bad_alloc e) throw e; catch() return p; 在默认情况下,new_handler的行为是抛出一个bad_alloc异常,因此上述循环只会执行一次。但如果我们不希望使用默认行为,可以自定义一个new_handler,并使用std:set_new_handler函数使其生效。在自定义的new_handler中,我们可以抛出异常,可以结束程序,也可以运行一些代码使得有可能有内存被空闲出来,从而下一次分配时也许会成功,也可以通过set_new_handler来安装另一个可能更有效的new_handler。例如: void MyNewHandler printf(New handler called!n); throw std:bad_alloc; std:set_new_handler(MyNewHandler); 这里new_handler程序在抛出异常之前会输出一句话。应该注意,在new_handler的代码里应该注意避免再嵌套有对new的调用,因为如果这里调用new再失败的话,可能会再导致对new_handler的调用,从而导致无限递归调用。这是我猜的,并没有尝试过。 在编程时我们应该注意到对new的调用是有可能有异常被抛出的,因此在new的代码周围应该注意保持其事务性,即不能因为调用new失败抛出异常来导致不正确的程序逻辑或数据结构的出现。例如: class SomeClass static int count; SomeClass public: static SomeClass* GetNewInstance count+; return new SomeClass; ; 静态变量count用于记录此类型生成的实例的个数,在上述代码中,如果因new分配内存失败而抛出异常,那么其实例个数并没有增加,但count变量的值却已经多了一个,从而数据结构被破坏。正确的写法是: static SomeClass* GetNewInstance SomeClass* p = new SomeClass; count+; return p; 这样一来,如果new失败则直接抛出异常,count的值不会增加。类似的,在处理线程同步时,也要注意类似的问题: void SomeFunc lock(someMutex); /加一个锁 delete p; p = new SomeClass; unlock(someMutex); 此时,如果new失败,unlock将不会被执行,于是不仅造成了一个指向不正确地址的指针p的存在,还将导致someMutex永远不会被解锁。这种情况是要注意避免的。 STL的内存分配与traits技巧 在STL原码剖析一书中详细分析了SGI STL的内存分配器的行为。与直接使用new operator不同的是,SGI STL并不依赖C+默认的内存分配方式,而是使用一套自行实现的方案。首先SGI STL将可用内存整块的分配,使之成为当前进程可用的内存,当程序中确实需要分配内存时,先从这些已请求好的大内存块中尝试取得内存,如果失败的话再尝试整块的分配大内存。这种做法有效的避免了大量内存碎片的出现,提高了内存管理效率。 为了实现这种方式,STL使用了placement new,通过在自己管理的内存空间上使用placement new来构造对象,以达到原有new operator所具有的功能。 template <class T1, class T2> inline void construct(T1* p, const T2& value) new(p) T1(value); 此函数接收一个已构造的对象,通过拷贝构造的方式在给定的内存地址p上构造一个新对象,代码中后半截T1(value)便是placement new语法中调用构造函数的写法,如果传入的对象value正是所要求的类型T1,那么这里就相当于调用拷贝构造函数。类似的,因使用了placement new,编译器不会自动产生调用析构函数的代码,需要手工的实现: template <class T> inline void destory(T* pointer) pointer->T; 与此同时,STL中还有一个接收两个迭代器的destory版本,可将某容器上指定范围内的对象全部销毁。典型的实现方式就是通过一个循环来对此范围内的对象逐一调用析构函数。如果所传入的对象是非简单类型,这样做是必要的,但如果传入的是简单类型,或者根本没有必要调用析构函数的自定义类型,那么再逐一调用析构函数是没有必要的,也浪费了时间。为此,STL使用了一种称为type traits的技巧,在编译器就判断出所传入的类型是否需要调用析构函数: template <class ForwardIterator> inline void destory(ForwardIterator first, ForwardIterator last) _destory(first, last, value_type(first); 其中value_type用于取出迭代器所指向的对象的类型信息,于是: template<class ForwardIterator, class T> inline void _destory(ForwardIterator first, ForwardIterator last, T*) typedef typename _type_traits<T>:has_trivial_destructor trivial_destructor; _destory_aux(first, last, trivial_destructor); /如果需要调用析构函数: template<class ForwardIterator> inline void _destory_aux(ForwardIterator first, ForwardIterator last, _false_type) for(; first < last; +first) destory(&*first); /因first是迭代器,*first取出其真正内容,然后再用&取地址 /如果不需要,就什么也不做: tempalte<class ForwardIterator> inline void _destory_aux(ForwardIterator first, ForwardIterator last, _true_type) 因上述函数全都是inline的,所以多层的函数调用并不会对性能造成影响,最终编译的结果根据具体的类型就只是一个for循环或者什么都没有。这里的关键在于_type_traits<T>这个模板类上,它根据不同的T类型定义出不同的has_trivial_destructor的结果,如果T是简单类型,就定义为_true_type类型,否则就定义为_false_type类型。其中_true_type、_false_type只不过是两个没有任何内容的类,对程序的执行结果没有什么意义,但在编译器看来它对模板如何特化就具有非常重要的指导意义了,正如上面代码所示的那样。_type_traits<T>也是特化了的一系列模板类: struct _true_type ; struct _false_type ; template <class T> struct _type_traits public: typedef _false _type has_trivial_destructor; ; template<> /模板特化 struct _type_traits<int> /int的特化版本 public: typedef _true_type has_trivial_destructor; ; /其他简单类型的特化版本 如果要把一个自定义的类型MyClass也定义为不调用析构函数,只需要相应的定义_type_traits<T>的一个特化版本即可: template<> struct _type_traits<MyClass> public: typedef _true_type has_trivial_destructor; ; 模板是比较高级的C+编程技巧,模板特化、模板偏特化就更是技巧性很强的东西,STL中的type_traits充分借助模板特化的功能,实现了在程序编译期通过编译器来决定为每一处调用使用哪个特化版本,于是在不增加编程复杂性的前提下大大提高了程序的运行效率。更详细的内容可参考STL源码剖析第二、三章中的相关内容。 带有的new和delete 我们经常会通过new来动态创建一个数组,例如: char* s = new char100; delete s; 严格的说,上述代码是不正确的,因为我们在分配内存时使用的是new,而并不是简单的new,但释放内存时却用的是delete。正确的写法是使用delete: delete s; 但是,上述错误的代码似乎也能编译执行,并不会带来什么错误。事实上,new与new、delete与delete是有区别的,特别是当用来操作复杂类型时。假如针对一个我们自定义的类MyClass使用new: MyClass* p = new MyClass10; 上述代码的结果是在堆上分配了10个连续的MyClass实例,并且已经对它们依次调用了构造函数,于是我们得到了10个可用的对象,这一点与Java、C#有区别的,Java、C#中这样的结果只是得到了10个null。换句话说,使用这种写法时MyClass必须拥有不带参数的构造函数,否则会发现编译期错误,因为编译器无法调用有参数的构造函数。 当这样构造成功后,我们可以再将其释放,释放时使用delete: delete p; 当我们对动态分配的数组调用delete时,其行为根据所申请的变量类型会有所不同。如果p指向简单类型,如int、char等,其结果只不过是这块内存被回收,此时使用delete与delete没有区别,但如果p指向的是复杂类型,delete会针对动态分配得到的每个对象调用析构函数,然后再释放内存。因此,如果我们对上述分配得到的p指针直接使用delete来回收,虽然编译期不报什么错误,但在运行时会给出一个Debug assertion failed提示。 到这里,我们很容易提出一个问题delete是如何知道要为多少个对象调用析构函数的?要回答这个问题,我们可以首先看一看new的重载。 class MyClass int a; public: MyClass printf("ctorn"); MyClass printf("dtorn"); ; void* operator new(size_t size) void* p = operator new(size); printf("calling new with size=%d address=%pn", size, p); return p; / 主函数 MyClass* mc = new MyClass3; printf("address of mc=%pn", mc); delete mc; 运行此段代码,得到的结果为: calling new with size=16 address=003A5A58 ctor ctor ctor address of mc=003A5A5C dtor dtor dtor 虽然对构造函数和析构函数的调用结果都在预料之中,但所申请的内存空间大小以及地址的数值却出现了问题。我们的类MyClass的大小显然是4个字节,并且申请的数组中有3个元素,那么应该一共申请12个字节才对,但事实上系统却为我们申请了16字节,并且在operator new返后我们得到的内存地址是实际申请得到的内存地址值加4的结果。也就是说,当为复杂类型动态分配数组时,系统自动在最终得到的内存地址前空出了4个字节,我们有理由相信这4个字节的内容与动态分配数组的长度有关。通过单步跟踪,很容易发现这4个字节对应的int值为0x00000003,也就是说记录的是我们分配的对象的个数。改变一下分配的个数然后再次观察的结果证实了我的想法。于是,我们也有理由认为new operator的行为相当于下面的伪代码: template <class T> T* New(int count) int size = sizeof(T) * count + 4; void* p = T:operator new(size); *(int*)p = count; T* pt = (T*)(int)p + 4); for(int i = 0; i < count; i+) new(&pti) T; return pt; 上述示意性的代码省略了异常处理的部分,只是展示当我们对一个复杂类型使用new来动态分配数组时其真正的行为是什么,从中可以看到它分配了比预期多4个字节的内存并用它来保存对象的个数,然后对于后面每一块空间使用placement new来调用无参构造函数,这也就解释了为什么这种情况下类必须有无参构造函数,最后再将首地址返回。类似的,我们很容易写出相应的delete的实现代码: template <class T> void Delete(T* pt) int count = (int*)pt)-1; for(int i = 0; i < count; i+) pti.T; void* p = (void*)(int)pt 4); T:operator delete(p); 由此可见,在默认情况下operator new与operator new的行为是相同的,operator delete与operator delete也是,不同的是new operator与new operator、delete operator与delete operator。当然,我们可以根据不同的需要来选择重载带有和不带有的operator new和delete,以满足不同的具体需求。 把前面类MyClass的代码稍做修改注释掉析构函数,然后再来看看程序的输出: calling new with size=12 address=003A5A58 ctor ctor ctor address of mc=003A5A58 这一次,new老老实实的申请了12个字节的内存,并且申请的结果与new operator返回的结果也是相同的,看来,是否在前面添加4个字节,只取决于这个类有没有析构函数,当然,这么说并不确切,正确的说法是这个类是否需要调用构造函数,因为如下两种情况下虽然这个类没声明析构函数,但还是多申请了4个字节:一是这个类中拥有需要调用析构函数的成员,二是这个类继承自需要调用析构函数的类。于是,我们可以递归的定义需要调用析构函数的类为以下三种情况之一: 1 显式的声明了析构函数的 2 拥有需要调用析构函数的类的成员的 3 继承自需要调用析构函数的类的 类似的,动态申请简单类型的数组时,也不会多申请4个字节。于是在这两种情况下,释放内存时使用delete或delete都可以,但为养成良好的习惯,我们还是应该注意只要是动态分配的数组,释放时就使用delete。 释放内存时如何知道长度 但这同时又带来了新问题,既然申请无需调用析构函数的类或简单类型的数组时并没有记录个数信息,那么operator delete,或更直接的说free是如何来回收这块内存的呢?这就要研究malloc返回的内存的结构了。与new类似的是,实际上在malloc申请内存时也多申请了数个字节的内容,只不过这与所申请的变量的类型没有任何关系,我们从调用malloc时所传入的参数也可以理解这一点它只接收了要申请的内存的长度,并不关系这块内存用来保存什么类型。下面运行这样一段代码做个实验: char *p = 0; for(int i = 0; i < 40; i += 4) char* s = new chari; printf("alloc %2d bytes, address=%p distance=%dn", i, s, s - p); p = s; 我们直接来看VC2005下Release版本的运行结果,DEBUG版因包含了较多的调试信息,这里就不分析了: alloc 0 bytes, address=003A36F0 distance=3815152 alloc 4 bytes, address=003A3700 distance=16 alloc 8 bytes, address=003A3710 distance=16 alloc 12 bytes, address=003A3720 distance=16 alloc 16 bytes, address=003A3738 distance=24 alloc 20 bytes, address=003A84C0 distance=19848 alloc 24 bytes, address=003A84E0 distance=32 alloc 28 bytes, address=003A8500 distance=32 alloc 32 bytes, address=003A8528 distance=40 alloc 36 bytes, address=003A8550 distance=40 每一次分配的字节数都比上一次多4,distance值记录着与上一次分配的差值,第一个差值没有实际意义,中间有一个较大的差值,可能是这块内存已经被分配了,于是也忽略它。结果中最小的差值为16字节,直到我们申请16字节时,这个差值变成了24,后面也有类似的规律,那么我们可以认为申请所得的内存结构是如下这样的: 从图中不难看出,当我们要分配一段内存时,所得的内存地址和上一次的尾地址至少要相距8个字节,那么我们可以猜想,这8个字节中应该记录着与这段所分配的内存有关的信息。观察这8个节内的内容,得到结果如下: 图中右边为每次分配所得的地址之前8个字节的内容的16进制表示,从图中红线所表示可以看到,这8个字节中的第一个字节乘以8即得到相临两次分配时的距离,经过试验一次性分配更大的长度可知,第二个字节也是这个意义,并且代表高8位,也就说前面空的这8个字节中的前两个字节记录了一次分配内存的长度信息,后面的六个字节可能与空闲内存链表的信息有关,在翻译内存时用来提供必要的信息。这就解答了前面提出的问题,原来C/C+在分配内存时已经记录了足够充分的信息用于回收内存,只不过我们平常不关心它罢了。

    注意事项

    本文(C++中new 的用法大全 详细.docx)为本站会员(牧羊曲112)主动上传,三一办公仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知三一办公(点击联系客服),我们立即给予删除!

    温馨提示:如果因为网速或其他原因下载失败请重新下载,重复下载不扣分。




    备案号:宁ICP备20000045号-2

    经营许可证:宁B2-20210002

    宁公网安备 64010402000987号

    三一办公
    收起
    展开