[C/C++]_[中级]_[delete 类对象指针的注意事项]
场景:
1. C++类有构造和析构函数,析构函数是在类对象被delete时(或局部变量自动销毁时)调用来释放资源。
2. C++类对象指针很多情况下需要赋值给void*通用指针来达到传输对象的目的,但是往往这种void*指针就是造成内存泄漏或程序错误的根源,
这就是为什么C++存在泛型的目的,它也是为了在编译时刻消除这种对象不确定性,避免delete或使用时的错误.
C 的析构里deleta data_, 看反汇编代码,有调用析构函数.
1. C++类有构造和析构函数,析构函数是在类对象被delete时(或局部变量自动销毁时)调用来释放资源。
2. C++类对象指针很多情况下需要赋值给void*通用指针来达到传输对象的目的,但是往往这种void*指针就是造成内存泄漏或程序错误的根源,
这就是为什么C++存在泛型的目的,它也是为了在编译时刻消除这种对象不确定性,避免delete或使用时的错误.
3. delete void*类型时,注意要强制转换为类类型才delete, 如 delete (A*) data_;
好了,看代码,以下代码有什么问题?
// test_class.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include <iostream> class A { public: A() { i = new int; } ~A() { delete i; } int* i; }; class B { public: B(void* data) { data_ = data; } ~B() { delete data_; } void* data_; }; template <class T> class C { public: C(T* data) { data_ = data; } ~C() { delete data_; } T* data_; }; void Wrong() { A *a = new A(); B b(a); //函数返回时A 的析构函数不会调用 } void Right() { A *a = new A(); C<A> c(a); //函数返回时A 的析构函数会调用 } int _tmain(int argc, _TCHAR* argv[]) { Wrong(); Right(); return 0; }
解析:
B 的析构里deleta data_, 看反汇编代码,并没有调用析构函数.
011D1643 mov eax,dword ptr [this] 011D1646 mov ecx,dword ptr [eax] 011D1648 mov dword ptr [ebp-0D4h],ecx 011D164E mov edx,dword ptr [ebp-0D4h] 011D1654 push edx 011D1655 call operator delete (11D1096h)
C 的析构里deleta data_, 看反汇编代码,有调用析构函数.
011D1883 mov eax,dword ptr [this] 011D1886 mov ecx,dword ptr [eax] 011D1888 mov dword ptr [ebp-0D4h],ecx 011D188E mov edx,dword ptr [ebp-0D4h] 011D1894 mov dword ptr [ebp-0E0h],edx 011D189A cmp dword ptr [ebp-0E0h],0 011D18A1 je C<A>::~C<A>+58h (11D18B8h) 011D18A3 push 1 011D18A5 mov ecx,dword ptr [ebp-0E0h] 011D18AB call A::`scalar deleting destructor" (11D102Dh) 011D18B0 mov dword ptr [ebp-0E8h],eax 011D18B6 jmp C<A>::~C<A>+62h (11D18C2h)
声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。