牛骨文教育服务平台(让学习变的简单)
博文笔记

[C/C++]_[中级]_[delete 类对象指针的注意事项]

创建时间:2015-05-07 投稿人: 浏览次数:5814
场景:
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)  



声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。