http://blog.csdn.net/infoworld/article/details/45560219

场景:
1. C++类有构造和析构函数,析构函数是在类对象被delete时(或局部变量自动销毁时)调用来释放资源。

2. C++类对象指针很多情况下需要赋值给void*通用指针来达到传输对象的目的,但是往往这种void*指针就是造成内存泄漏或程序错误的根源,

这就是为什么C++存在泛型的目的,它也是为了在编译时刻消除这种对象不确定性,避免delete或使用时的错误.

3. delete void*类型时,注意要强制转换为类类型才delete, 如 delete (A*) data_;

好了,看代码,以下代码有什么问题?

#include <iostream>
using namespace std;
class A
{
public:
    A()
    {
        i = new int;
        cout << "A()! " << "lenI:"<<sizeof(i) << endl;
    }
    ~A()
    {
        delete i;
        cout << "~A ! " << endl;
    }
    int* i;
};
class B
{
public:
    B(void* data)
    {
        data_ = data;
        cout << "B(void* data) ! " << endl;
    }
    ~B()
    {
        delete data_;
        cout << "~B()! " << endl;
    }
    void* data_;
};
template <class T>
class C
{
public:
    C(T* data)
    {
        data_ = data;
        cout << "C(T* data)! " << endl;
    }
    ~C()
    {
        delete data_;
        cout << "~C()! " << endl;
    }
    T* data_;
};
void Wrong()
{
    A *a = new A();
    B b(a); //函数返回时A 的析构函数不会调用
    cout << "Wrong()! " << endl;
}
void Right()
{
    A *a = new A();
    C<A> c(a); //函数返回时A 的析构函数会调用
    cout << "Right)! " << endl;
}
int main(int argc, char* argv[])
{
    Wrong();
    Right();
    return 0;
}


  1. // test_class.cpp : 定义控制台应用程序的入口点。
  2. //
  3. #include "stdafx.h"
  4. #include <iostream>
  5. class A
  6. {
  7. public:
  8. A()
  9. {
  10. i = new int;
  11. }
  12. ~A()
  13. {
  14. delete i;
  15. }
  16. int* i;
  17. };
  18. class B
  19. {
  20. public:
  21. B(void* data)
  22. {
  23. data_ = data;
  24. }
  25. ~B()
  26. {
  27. delete data_;
  28. }
  29. void* data_;
  30. };
  31. template <class T>
  32. class C
  33. {
  34. public:
  35. C(T* data)
  36. {
  37. data_ = data;
  38. }
  39. ~C()
  40. {
  41. delete data_;
  42. }
  43. T* data_;
  44. };
  45. void Wrong()
  46. {
  47. A *a = new A();
  48. B b(a); //函数返回时A 的析构函数不会调用
  49. }
  50. void Right()
  51. {
  52. A *a = new A();
  53. C<A> c(a); //函数返回时A 的析构函数会调用
  54. }
  55. int _tmain(int argc, _TCHAR* argv[])
  56. {
  57. Wrong();
  58. Right();
  59. return 0;
  60. }

解析:

B 的析构里deleta data_, 看反汇编代码,并没有调用析构函数.

  1. 011D1643  mov         eax,dword ptr [this]
  2. 011D1646  mov         ecx,dword ptr [eax]
  3. 011D1648  mov         dword ptr [ebp-0D4h],ecx
  4. 011D164E  mov         edx,dword ptr [ebp-0D4h]
  5. 011D1654  push        edx
  6. 011D1655  call        operator delete (11D1096h)

C 的析构里deleta data_, 看反汇编代码,有调用析构函数.

      1. 011D1883  mov         eax,dword ptr [this]
      2. 011D1886  mov         ecx,dword ptr [eax]
      3. 011D1888  mov         dword ptr [ebp-0D4h],ecx
      4. 011D188E  mov         edx,dword ptr [ebp-0D4h]
      5. 011D1894  mov         dword ptr [ebp-0E0h],edx
      6. 011D189A  cmp         dword ptr [ebp-0E0h],0
      7. 011D18A1  je          C<A>::~C<A>+58h (11D18B8h)
      8. 011D18A3  push        1
      9. 011D18A5  mov         ecx,dword ptr [ebp-0E0h]
      10. 011D18AB  call        A::`scalar deleting destructor' (11D102Dh)
      11. 011D18B0  mov         dword ptr [ebp-0E8h],eax
      12. 011D18B6  jmp         C<A>::~C<A>+62h (11D18C2h)

delete 类对象指针的注意事项]的更多相关文章

  1. mfc 类对象指针

    类对象指针 一.类对象指针定义 Tdate d1; Tdate *p1=&d1; Tdate *p2=(Tdate *)malloc(sizeof(Tdate)); 二.类对象指针使用 int ...

  2. c++类对象 指针区别

    class Test{ public: int a; Test(){ a = ; } }; int main1() { Test* t1 = new Test(); t1->a = ; Test ...

  3. class中new与未new的区别 类对象占用空间--转载

    转载自http://blog.sina.com.cn/shuiwuhendeboke    颗颗的博客 (1)作用域不同 不用new:作用域限制在定义类对象的方法中,当方法结束时,类对象也被系统释放了 ...

  4. C++用new和不用new创建类对象区别

    new创建类对象,使用完后需使用delete删除,跟申请内存类似.所以,new有时候又不太适合,比如在频繁调用场合,使用局部new类对象就不是个好选择,使用全局类对象或一个经过初始化的全局类指针似乎更 ...

  5. Python全栈--9.1--面向对象进阶-super 类对象成员--类属性- 私有属性 查找源码类对象步骤 类特殊成员 isinstance issubclass 异常处理

    上一篇文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使用(可以讲多函数中公用的变量封装到对象中) 对象 ...

  6. python基础-9.1 面向对象进阶 super 类对象成员 类属性 私有属性 查找源码类对象步骤 类特殊成员 isinstance issubclass 异常处理

    上一篇文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使用(可以讲多函数中公用的变量封装到对象中) 对象 ...

  7. 对象析构谈—— delete this 的使用及注意事项

    this对象是必须是用 new操作符分配的(而不是用new[],也不是用placement new,也不是局部对象,也不是global对象): delete this后,不能访问该对象任何的成员变量及 ...

  8. C++ 类对象和 指针的区别

    C++ 类对象和 指针的区别 C++ 类对象和 指针的区别 转自:http://blog.csdn.net/ym19860303/article/details/8557746 指针的情况 class ...

  9. c++中的类的对象与类的指针

    以上内容来自:http://wenku.baidu.com/link?url=haeRBhswlEcqddk48uW8YVMsdFNWsllimn_dzUYchb6G9NdT4pqgluCpnLQId ...

随机推荐

  1. Jfreechart 乱码

    整个图标分成三部分chart   title,chart 的plot还有chart的   legend三个部分需要对他们分别设置字体就对了. 先看解决方法( 把这几个全部设置了,都搞定了就可以了): ...

  2. POJ2677 Tour(DP+双调欧几里得旅行商问题)

    Tour Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3929   Accepted: 1761 Description ...

  3. 文件流StreamReader和StreamWriter的使用

    using (StreamReader sr = new StreamReader(@"C:\Users\shuai\Desktop\文件流读取.txt", Encoding.De ...

  4. boost(barrier)

    barrier:栅栏的意思,当barrier bar(3),这三个线程会放到栅栏中,等第三个线程执行时一起唤醒,然后返回 barrier barrier类的接口定义如下: class barrier ...

  5. C语言代码优化(转)

    .选择合适的算法和数据结构 选择一种合适的数据结构很重要,如果在一堆随机存放的数中使用了大量的插入和删除指令,那使用链表要快得多.数组与指针语句具有十分密切的关系,一般来说,指针比较灵活简洁,而数组则 ...

  6. javascript正则——贪婪匹配

    熟悉正则的朋友都知道,正则的匹配有“贪婪”和“非贪婪”之分. “贪婪”匹配是尽可能多的匹配: 对于字符串‘aaaa’, /a+/匹配整个字符串,而非贪婪匹配/a+?/匹配的是整个字符串的第一个‘a’, ...

  7. HDOJ 1536 S-Nim

    S-Nim Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submi ...

  8. java笔记--关于克隆技术

    关于克隆 --如果朋友您想转载本文章请注明转载地址"http://www.cnblogs.com/XHJT/p/3884817.html"谢谢-- 1.假克隆 如: ObjectA ...

  9. Android Bitmap 全面解析(四)图片处理效果对比 ...

    对比对象: UIL Volley 官方教程中的方法(此系列教程一里介绍的,ImageLoader的处理方法和官方的差不多) -------------------------------------- ...

  10. 虚拟机里面安装Openfiler 2.99

    简介 Openfiler 由rPath Linux驱动,它是一个基于浏览器的免费网络存储管理实用程序,可以在单一框架中提供基于文件的网络连接存储 (NAS) 和基于块的存储区域网 (SAN).Open ...