智能指针
当类中有指针成员时,一般来说有两种方式管理指针成员:
- 一种是采用值型的方式管理,每个类对象都保留一份指针指向的对象的拷贝;
- 另一种则是使用智能指针(smart pointer),从而实现指针指向的对象的共享。
智能指针的一种通用实现技术是使用引用技术(reference count)。智能指针类将一个计数器与类指向的对象相关量,引用计数跟踪该类有多少个对象的指针指向同一个对象。
原理
每次创建类的新对象时,初始化指针并将引用计数置位1;当对象作为另一对象的副本而创建时,拷贝构造函数拷贝拷贝并增加与之相应的引用计数:
对一个对象进行赋值,赋值操作符减少左操作数所指对象的引用计数(如果引用计数变为0,则删除对象),并增加右操作数所指对象的引用计数;
调用析构函数时,析构函数减少引用计数(如果引用计数减至0,则删除基础对象)。
STL内的智能指针
STL中一共提供了四类智能指针:
- auto_ptr
- unique_ptr
- shared_ptr
- weak_ptr
其中,auto_ptr是C++98所提供的解决方案,C++11已经摒弃了,并提出了unique_ptr作为替代方案。
当然,虽然shared_ptr被遗弃了,但实际项目中仍然可以使用,但是建议使用较新的unique_ptr,因为unique_ptr比auto_ptr更加安全。
shared_ptr和weak_ptr则是C++11从准标准库Boost所引入的两种只能指针。
此外,Boost库还提出了boost::scoped_ptr、boost::scoped_array、boost::intrusive_ptr 等智能指针,虽然尚未得到C++标准采纳,但是实际开发工作中可以使用。
智能指针的作用
在C++中,使用堆内存是非常频繁的操作,堆内存的申请和释放都需要由程序员自己去管理。程序员自己管理堆内存可以提高了程序的效率,但是整体来说堆内存的管理是麻烦的,C++11中引入了智能指针的概念,方便管理堆内存。使用普通指针,容易造成堆内存泄露(忘记释放),二次释放,程序发生异常时内存泄露等问题等,使用智能指针能更好的管理堆内存。
对于智能指针而言:
- 从简单的层次理解,智能指针利用了**RAII(资源获得即初始化)**的技术对普通的指针进行封装。这就使得智能指针实际上是一个对象,行为表现的却像一个指针。
- 智能指针的作用是防止忘记调用delete释放内存和程序异常的进入catch块忘记释放内存。另外指针的释放时机也是非常有考究的,多次释放同一个指针会造成程序崩溃,这些都可以通过智能指针来解决。
- 智能指针还有一个作用是把值语义转换成引用语义。C++和Java有一处最大的区别在于语义不同,在Java里面下列代码:
Animal a = new Animal();
Animal b = a;
这里其实只生成了一个对象,a和b仅仅是把持对象的引用而已。但在C++中不是这样:
Animal a;
Animal b = a;
这里是生成了两个对象。