mozilla开源项目学习----智能指针(AutoPtr&AutoArrayPtr)(上)
用过c++朋友一定都忘记释放过动态分配的内存。内存释放是一个很头痛的问题,一不小心就会造成内存泄漏。比如说,一个函数中分配了内存,但这个函数还有很多出口(return),我们不得不在每个出口去释放内存。这是一个很费时费力的的工作。但如果用了智能指针,在函数结束时,会智能的释放内存。这节省了程序员的工作,也减少程序员犯错。
在mozilla开源项目里用到了智能指针nsAutoPtr和nsAutoArrayPtr(两个c++模板类)。这两个类会在类生命周期结束时释放所管理的内存。在代码中使用这类时,就不必担心内存释放。
AutoPtr:
Ptr是nsAutoPtr内部的类。注释说Ptr类阻止隐晦复制构造函数。
class Ptr
{
public:
Ptr(T* aPtr) : mPtr(aPtr)
{
}
operator T*() const
{
return mPtr;
}
private:
T* mPtr;
};
Ptr重载了隐式类转换,直接返回Ptr::mPtr成员值;
Ptr构造函数有一个参数,用于初始化Ptr::mPtr;
nsAutoPtr有三个构造函数。
nsAutoPtr的默认构造函数只将nsAutoPtr::mRawPtr初始化成0;
nsAutoPtr() : mRawPtr(0)
{
}
nsAutoPtr(Ptr aRawPtr) : mRawPtr(aRawPtr)
{
}
此构造函数使用Ptr类构造类初始化nsAutoPtr::mRawPtr指针。
nsAutoPtr(nsAutoPtr<T>& aSmartPtr)
: mRawPtr(a.SmartPtr.forget())
{
}
当使用智能指针A初始化另一个智能指针B时,被初始化的智能指针B将拥有智能指针B所管理的内存,而B将失败对内存管理。这个设计理由是,只有一个nsAutoPtr类管理这个内存。假设有两个nsAutoPtr A和b管理一个内存,当其中一个nsAutoPtr A生命周期结束,内存将被释放,那么B将指向一个非法内存地址。
nsAutoPtr<T>& operator=(T* rhs)
{
assign(rhs);
return *this;
}
此函数重载了右值为T* 的=运算符,nsAutoPtr首先释放原先管理内存,再管理rhs所指向内存。这避免泄漏原来管理的那内存。
nsAutoPtr<T>& operator=(nsAutoPtr& rhs)
{
assign(rhs.forget());
return *this;
}
此成员函数重载了右值为nsAutoPtr&的=运算符。 左值nsAutoPtr将释放管理的内存,再接管右值nsAutoPtr管理的内存。而右值nsAutoPtr将失去内存的管理权。这保证了这个内存只被一个nsAutoPtr管理。
operator T*() const
{
return get();
}
此函数重载了隐式类型转换。这主要是方便智能指针使用。例如 nsAutoPtr<int> a = new int; int * b = a;
b = a;这条语句不针产生编译错误,b 就是等于a的实例成员mRawPtr指向的内存。
T* operator->() const
{
NS_PRECONDITION(m_RawPtr != 0, “You can’t dereference a NULL nsAutoPtr with operator->(). ");
return get();
}
此成员函数重载->运算符。对nsAutoPtr<T> 类的操作转换成对T类型的操作。这也将方便nsAutoPtr的使用,就像是直接操作T类型的变量。
例如:
class A
{
public:
void FunctionA() {};
int nVa;
};
int main()
{
nsAutoPtr<A> a = new A;
a->FunctionA();
a->nVa;
return 0;
}
T& operator*() const
{
NS_PRECONDITION(mRawPtr != 0, “You can’t derefence a NULL nsAutoPtr width operator*().”);
return *get();
}
此函数重载了“*”运算。将对nsAutoPtr的运算转换成对nsAutoPtr<T>::mRawPtr的运算。
nsAutoPtr基本上对运算符都做了重载。将对自身的运算都转换到对成员T*的运算。这要在添加了nsAutoPtr的保护后,不会影响程序员的操作。而对nsAutoPtr<T>变量的操作,都写了另一些函数来实现, 这对nsAutoPtr<T>变量的操作是变麻烦了。但相对于带来的好处,这点麻烦还是承受的起的。
小弟只是肤浅了表达下自己的理解,希望多多指正,受教。