[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)
声明:该文观点仅代表作者本人,入门客AI创业平台信息发布平台仅提供信息存储空间服务,如有疑问请联系rumenke@qq.com。
