C++进阶(智能指针)
正文:
C++步调设想中运用堆内存是很是频繁的收配,堆内存的申请和开释都由步调员原人打点。步调员原人打点堆内存可以进步了步调的效率,但是整体来说堆内存的打点是省事的,C++11中引入了智能指针的观念,便捷打点堆内存。运用普通指针,容易组成堆内存泄露(忘记开释),二次开释,步调发作异样时内存泄露等问题等,运用智能指针能更好的打点堆内存。
从较浅的层面看,智能指针是操做了一种叫作RAII(资源获与即初始化)的技术对普通的指针停行封拆,那使得智能指针原量是一个对象,止为暗示的却像一个指针。
智能指针的做用是避免忘记挪用delete开释内存和步调异样的进入catch块忘记开释内存。此外指针的开释时机也是很是有讲究的,多次开释同一个指针会组成步调解体,那些都可以通过智能指针来处置惩罚惩罚。
智能指针次要用于打点正在堆上分配的内存,它将普通的指针封拆为一个栈对象。当栈对象的保留周期完毕后,会正在析构函数中开释掉申请的内存,从而避免内存泄漏。
智能指针的做用是打点一个指针,因为存正在以下那种状况:申请的空间正在函数完毕时忘记开释,组成内存泄漏。运用智能指针可以很急流平上的防行那个问题,因为智能指针是一个类,当超出了类的真例对象的做用域时,会主动挪用对象的析构函数,析构函数会主动开释资源。所以智能指针的做用本理便是正在函数完毕时主动开释内存空间,不须要手动开释内存空间。
智能指针的运用智能指针正在C++11版原之后供给,包孕正在头文件< memory>中:shared_ptr、unique_ptr、weak_ptr。(留心:auto_ptr是一种存正在缺陷的智能指针,正在C++11中曾经被进用了)
shared_ptr允很多个指针指向同一个对象,unique_ptr则“独占”所指向的对象。范例库还界说了一种名为weak_ptr的随同类,它是一种弱引用,指向shared_ptr所打点的对象。
RAII(1)根柢观念
①RAII(Resource Acquisition Is Initialization)是一种操做对象生命周期来控制步调资源(如内存、文件句柄、网络连贯、互斥质等等)的简略技术。
②正在对象结构时获与资源,接着控制对资源的会见使之正在对象的生命周期内始末保持有效,最后正在对象析构的时候开释资源。借此, 咱们真际上把打点一份资源的义务托管给了一个对象 。那种作法有两大好处
不须要显式地开释资源
给取那种方式,对象所需的资源正在其生命期内始末保持有效
(2)代码模拟
真现智能指针时须要思考以下三个方面的问题:
正在对象结构时获与资源,正在对象析构的时候开释资源,操做对象的生命周期来控制步调资源,即RAII特性
对*和->运算符停行重载,使得该对象具有像指针一样的止为
智能指针对象的拷贝问题
// RAII // 用起来像指针一样 template<class T> class SmartPtr { public: SmartPtr(T* ptr) :_ptr(ptr) {} ~SmartPtr() { cout << "delete:" << _ptr << endl; delete _ptr; } // 像指针一样运用 T& operator*() { return *_ptr; } T* operator->() { return _ptr; } priZZZate: T* _ptr; };(3)为什么要处置惩罚惩罚智能指针对象的拷贝问题
应付当前真现的SmartPtr类,假如用一个SmartPtr对象来拷贝结构另一个SmartPtr对象,或是将一个SmartPtr对象赋值给另一个SmartPtr对象,都会招致步调解体
int main() { SmartPtr<int> sp1(new int); SmartPtr<int> sp2(sp1); //拷贝结构 SmartPtr<int> sp3(new int); SmartPtr<int> sp4(new int); sp3 = sp4; //拷贝赋值 return 0; }编译器默许生成的拷贝结构函数对内置类型完成值拷贝(浅拷贝),因而用sp1拷贝结构sp2后,相当于那sp1和sp2打点了同一块内存空间,当sp1和sp2析构时就会招致那块空间被开释两次。
编译器默许生成的拷贝赋值函数对内置类型也是完成值拷贝(浅拷贝),因而将sp4赋值给sp3后,相当于sp3和sp4打点的都是本来sp3打点的空间,当sp3和sp4析构时就会招致那块空间被开释两次,并且还会招致sp4本来打点的空间没有获得开释。
须要留心的是,智能指针便是要模拟本生指针的止为,当咱们将一个指针赋值给另一个指针时,宗旨便是让那两个指针指向同一块内存空间,所以那里原就应当停行浅拷贝,但单杂的浅拷贝又会招致空间被多次开释,因而依据处置惩罚惩罚智能指针拷贝问题方式的差异,从而衍生出了差异版原的智能指针。
unique_ptr 本理和运用unique_ptr“惟一”领有其所指对象,同一时刻只能有一个unique_ptr指向给定对象(通过制行拷贝语义、只要挪动语义来真现)。它应付防行资源泄露(譬喻“以new创立对象后因为发作异样而忘记挪用delete”)出格有用。
相比取本始指针,unique_ptr用于其RAII的特性,使得正在显现异样的状况下,动态资源能获得开释。unique_ptr指针自身的生命周期:从unique_ptr指针创立时初步,曲到分隔做用域。分隔做用域时,若其指向对象,则将其所指对象销誉(默许运用delete收配符,用户可指定其余收配)。
unique_ptr指针取其所指对象的干系:正在智能指针生命周期内,可以扭转智能指针所指对象,如创立智能指针时通过结构函数指定、通过reset办法从头指定、通过release办法开释所有权、通过挪动语义转移所有权。
示例:
注明:C++有一个范例库函数moZZZe(),让你能够将一个unique_ptr赋给另一个。只管转移所有权后还是有可能显现本有指针挪用(挪用就解体)的状况。但是那个语法能强调你是正在转移所有权,让你明晰的晓得原人正在作什么,从而不变挪用本有指针。
模拟真现 namespace XM { template<class T> class unique_ptr { public: unique_ptr(T* ptr = nullptr) : _ptr(ptr) {} ~unique_ptr() { if (_ptr) delete _ptr; } T& operator*() { return *_ptr; } T* operator->() { return _ptr; } priZZZate: // C++98防拷贝的方式:只声明不真现+声明成私有 unique_ptr(const unique_ptr<T>& sp); unique_ptr& operator=(const unique_ptr<T>& sp); // C++11防拷贝的方式:delete unique_ptr(const unique_ptr<T>& sp) = delete; unique_ptr& operator=(const unique_ptr<T>& sp) = delete; priZZZate: T* _ptr; }; } share_ptr 本理和运用C++ 11中最罕用的智能指针类型为shared_ptr。从名字share就可以看出了资源可以被多个指针共享,它运用计数机制来讲明资源被几多个指针共享。可以通过成员函数use_count()来查察资源的所有者个数。除了可以通过new来结构,还可以通过传入auto_ptr, unique_ptr, weak_ptr来结构。当咱们挪用release()时,当前指针会开释资源所有权,计数减一。当计数就是0时,资源会被开释。
shared_ptr的本理:是通过引用计数的方式来真现多个shared_ptr对象之间共享资源。
shared_ptr正在其内部,给每个资源都维护了着一份计数,用来记录该份资源被几多个对象共享。
正在对象被销誉时(也便是析构函数挪用),就注明原人不运用该资源了,对象的引用计数减一。
假如引用计数是0,就注明原人是最后一个运用该资源的对象,必须开释该资源;假如不是0,就注明除了原人另有其余对象正在运用该份资源,不能开释该资源,否则其余对象就成野指针了。
留心事项:
初始化。智能指针是个模板类,可以指定类型,传入指针通过结构函数初始化。也可以运用make_shared函数初始化。不能将指针间接赋值给一个智能指针,一个是类,一个是指针。譬喻:std::shared_ptr< int> p4 = new int(1);的写法是舛错的!
拷贝和赋值。拷贝使得对象的引用计数删多1,赋值使得本对象引用计数减1,当计数为0时,主动开释内存。厥后指向的对象引用计数加1,指向厥后的对象。
get函数获与本始指针。
不要用一个本始指针初始化多个shared_ptr,否则会组成二次开释同一内存。
防行循环引用。shared_ptr的一个最大的陷阱是循环引用,循环引用会招致堆内存无奈准确开释,招致内存泄漏。循环引用正在weak_ptr中引见。
成员函数:
use_count 返回引用计数的个数;
unique 返回能否是独占所有权(use_count 为 1);
swap 替换两个 shared_ptr 对象(即替换所领有的对象);
reset 放弃内部对象的所有权或领有对象的变更, 会惹起本有对象的引用计数的减少;
get 返回内部对象(指针), 由于曾经重载了()办法, 因而和间接运用对象是一样的。
示例:
class A { public: int _a = 10; ~A() { cout << "~A()" << endl; } }; ZZZoid test() { shared_ptr<A> sp(new A); shared_ptr<A> sp2(new A); shared_ptr<A> sp3(sp2);//ok sp3 = sp;//ok sp->_a = 100; sp2->_a = 1000; sp3->_a = 10000; cout << sp->_a << endl; cout << sp2->_a << endl; cout << sp3->_a << endl; }运止结果如下:
咱们发现申请几多多资源就会开释几多多资源,此时的sp和sp3共享一份资源,批改sp3也就相就是批改了sp。所以最末都会打印10000。这共享了一份资源,是如何真现资源只开释一次呢?----引用计数
咱们可以通过shared_ptr供给的接口use_count()来查察,当前有几多多个智能指针来打点同一份资源
ZZZoid test() { shared_ptr<A> sp(new A); cout << sp.use_count() << endl;//1 shared_ptr<A> sp2(sp); cout << sp.use_count() << endl;//2 cout << sp2.use_count() << endl;//2 shared_ptr<A> sp3(new A); cout << sp.use_count() << endl;//2 cout << sp2.use_count() << endl;//2 cout << sp3.use_count() << endl;//1 sp3 = sp; sp3 = sp2; cout << sp.use_count() << endl;//3 cout << sp2.use_count() << endl;//3 cout << sp3.use_count() << endl;//3 }运止截图:之所以中间会有调析构函数,是因为当sp3指向sp时,sp3的引用计数为0,则会挪用析构函数来开释资源。此时sp创立的资源就有3个指智能指针来打点
图解:
正在真现时,咱们应当确保一个资源只对应一个计数器,而不是每个智能指针都有各自的计数器。所以咱们可以将资源和计数器绑定正在一起,此时指向同一份资源的智能指针,会见的也都是同一个计数器(背面会评释)
模拟真现正在shared_ptr类中删多一个成员变质count,默示智能指针对象打点的资源对应的引用计数。
正在结构函数中获与资源,并将该资源对应的引用计数设置为1,默示当前只要一个对象正在打点那个资源。
正在拷贝结构函数中,取传入对象一起打点它打点的资源,同时将该资源对应的引用计数++。
正在拷贝赋值函数中,先将当前对象打点的资源对应的引用计数--(假如减为0则须要开释),而后再取传入对象一起打点它打点的资源,同时须要将该资源对应的引用计数++。
正在析构函数中,将打点资源对应的引用计数--,假如减为0则须要将该资源开释。
对*和->运算符停行重载,使shared_ptr对象具有指针一样的止为。
namespace XM { template<class T> class shared_ptr { public: shared_ptr(T* ptr) :_ptr(ptr) , _pRefCount(new int(1)) {} shared_ptr(const shared_ptr<T>& sp) :_ptr(sp._ptr) , _pRefCount(sp._pRefCount) { ++(*_pRefCount); } shared_ptr<T>& operator=(const shared_ptr<T>& sp) { if (_ptr != sp._ptr) { if (--(*_pRefCount) == 0) { delete _ptr; delete _pRefCount; } _ptr = sp._ptr; _pRefCount = sp._pRefCount; ++(*_pRefCount); } return *this; } ~shared_ptr() { if (--(*_pRefCount) == 0 && _ptr) { cout << "delete:" << _ptr << endl; delete _ptr; delete _pRefCount; //_ptr = nullptr; //_pRefCount = nullptr; } } int use_count() const { return *_pRefCount; } // 像指针一样运用 T& operator*() { return *_ptr; } T* operator->() { return _ptr; } priZZZate: T* _ptr; int* _pRefCount; }; }考虑一个问题:为什么引用计数要放正在堆区?★
①首先,shared_ptr中的引用计数count不能单杂的界说成一个int类型的成员变质,因为那就意味着每个shared_ptr对象都有一个原人的count成员变质,而当多个对象要打点同一个资源时,那几多个对象应当用到的是同一个引用计数。
②其次,shared_ptr中的引用计数count也不能界说成一个静态的成员变质,因为静态成员变质是所有类型对象共享的,那会招致打点雷同资源的对象和打点差异资源的对象用到的都是同一个引用计数。
③而假如将shared_ptr中的引用计数count界说成一个指针,当一个资源第一次被打点时就正在堆区斥地一块空间用于存储其对应的引用计数,假如有其余对象也想要打点那个资源,这么除了将那个资源给它之外,还须要把那个引用计数也给它。
④那时打点同一个资源的多个对象会见到的便是同一个引用计数,而打点差异资源的对象会见到的便是差异的引用计数了,相当于将各个资源取其对应的引用计数停行了绑定。
⑤但同时须要留心,由于引用计数的内存空间也是正在堆上斥地的,因而当一个资源对应的引用计数减为0时,除了须要将该资源开释,还须要将该资源对应的引用计数的内存空间停行开释。
线程安宁问题咱们真现的shared_ptr智能指针正在多线程的场景下其真是存正在线程安宁问题的----引用计数器指针是一个共享变质,多个线程停行批改时会招致计数器凌乱。招致资源提早被开释大概会孕育发作内存泄漏问题
咱们来看看一下代码
①通过实验结果可知,假如share_ptr不加锁正在多线程的状况下是不安宁的,正在pRefCount ++,- - 时 可能显现舛错
②智能指针对象中引用计数是多个智能指针对象共享的,两个线程中智能指针的引用计数同时++或--,那个收配不是本子的,引用计数本来是1,++了两次,可能还是2.那样引用计数就错乱了。会招致资源未开释大概步调解体的问题。所以只能指针中引用计数++、--是须要加锁的,也便是说引用计数的收配是线程安宁的。
③智能指针打点的对象寄存正在堆上,两个线程中同时去会见,会招致线程安宁问题
④那里智能指针会见打点的资源,不是线程安宁的;对Date的成员 ++ , 所以咱们看看那些值两个线程++了2n次,但是最末看到的结果,并一定是加了2n ; 为了担保线程安宁还要手动加锁
shared_ptr智能指针是线程安宁的吗?
是的,引用计数的加减是加锁护卫的。但是指向的资源不是线程安宁的,须要原人管
指向堆上资源的线程安宁问题是会见的人办理的,智能指针不论,也管不了; 引用计数的线程安宁问题,是智能指针要办理的
模拟线程安宁的代码 , 引用计数加锁
①要处置惩罚惩罚引用计数的线程安宁问题,素量便是要让对引用计数的自删和自减收配变为一个本子收配,因而可以对引用计数的收配停行加锁护卫,也可以用本子类atomic对引用计数停行封拆,那里以加锁为例
正在shared_ptr类中新删互斥锁成员变质,为了让打点同一个资源的多个线程会见到的是同一个互斥锁,打点差异资源的线程会见到的是差异的互斥锁,因而互斥锁也须要正在堆区创立。
正在挪用拷贝结构函数和拷贝赋值函数时,除了须要将对应的资源和引用计数交给当前对象打点之外,还须要将对应的互斥锁也交给当前对象。
当一个资源对应的引用计数减为0时,除了须要将对应的资源和引用计数停行开释,由于互斥锁也是正在堆区创立的,因而还须要将对应的互斥锁停行开释。
为了简化代码逻辑,可以将拷贝结构函数和拷贝赋值函数中引用计数的自删收配提与出来,封拆成AddRef函数,将拷贝赋值函数和析构函数中引用计数的自减收配提与出来,封拆成Release函数,那样就只须要对AddRef和Release函数停行加锁护卫便可。
namespace XM { template<class T> class shared_ptr { public: shared_ptr(T* ptr) :_ptr(ptr) , _pRefCount(new int(1)) ,_pmtV(new muteV) {} shared_ptr(const shared_ptr<T>& sp) :_ptr(sp._ptr) , _pRefCount(sp._pRefCount) ,_pmtV(sp._pmtV) { AddRef(); } shared_ptr<T>& operator=(const shared_ptr<T>& sp) { //if (this != &sp) 那样判断不太好,避免原人给原人赋值应当判断指针的值能否雷同 if (_ptr != sp._ptr) { Release(); _ptr = sp._ptr; _pRefCount = sp._pRefCount; _pmtV = sp._pmtV; AddRef(); } return *this; } ~shared_ptr() { Release(); } T* get() const { return _ptr; } int use_count() { return *_pRefCount; } T& operator*() { return *_ptr; } T* operator->() { return _ptr; } priZZZate: ZZZoid Release() //开释资源 { _pmtV->lock(); bool flag = false; if (--(*_pRefCount) == 0 && _ptr) { delete _ptr; delete _pRefCount; flag = true; //锁不能正在那里开释,因为背面要解锁 } _pmtV->unlock(); if (flag == true) { delete _pmtV; } } ZZZoid AddRef() //删多计数 { _pmtV->lock(); ++(*_pRefCount); _pmtV->unlock(); } priZZZate: T* _ptr; int* _pRefCount; muteV* _pmtV; }; }小结:
正在Release函数中,当引用计数被减为0时须要开释互斥锁资源,但不能正在临界区中开释互斥锁,因为背面还须要停行解锁收配,因而代码中借助了一个flag变质,通过flag变质来判断解锁后开释须要开释互斥锁资源。
shared_ptr只须要担保引用计数的线程安宁问题,而不须要担保打点的资源的线程安宁问题,就像本生指针打点一块内存空间一样,本生指针只须要指向那块空间,而那块空间的线程安宁问题应当由那块空间的收配者来担保
循环引用shared_ptr其真也存正在一些小问题,也便是循环引用问题
#include<iostream> #include<memory> #include<string> using namespace std; class A; class B; class A { public: shared_ptr<B> bptr; ~A() { cout << "class Ta is disstruct" << endl; } }; class B { public: shared_ptr<A>aptr; ~B() { cout << "class Tb is disstruct" << endl; } }; ZZZoid testPtr() { shared_ptr<A>ap(new A); shared_ptr<B>bp(new B); cout << "ap的引用计数" << ap.use_count() << endl;//ap的引用计数1 cout << "bp的引用计数" << bp.use_count() << endl;//bp的引用计数1 ap->bptr = bp; bp->aptr = ap; cout << "ap的引用计数" << ap.use_count() << endl;//ap的引用计数2 cout << "bp的引用计数" << bp.use_count() << endl;//bp的引用计数2 } int main() { testPtr(); return 0; }咱们可以用图来了解一下上述步调智能指针引用干系:
共享智能指针ap指向A的真例对象,内存引用计数+1,B的真例对象里面的成员aptr被ap赋值,所以aptr取ap怪异指向同一块内存,该内存引用计数变成2;同理指向B对象的也有两个共享智能指针,其引用计数也为2。
当函数完毕时,ap,bp两个共享智能指针分隔做用域,引用计数均减为1,正在那种状况下不会增除智能指针所打点的内存,招致A,B的真例对象不能被析构,最末组成内存泄漏,如图:
循环引用的处置惩罚惩罚方式 weak_ptrshare_ptr尽管曾经很好用了,但是有一点share_ptr智能指针还是有内存泄露的状况,当两个对象互相运用一个shared_ptr成员变质指向对方,会组成循环引用,使引用计数失效,从而招致内存泄漏。
weak_ptr是为了共同shared_ptr而引入的一种智能指针,因为它不具有普通指针的止为,没有重载operator*和->,它的最大做用正在于辅佐shared_ptr工做,像旁不雅观者这样不雅视察资源的运用状况。weak_ptr可以从一个shared_ptr大概另一个weak_ptr对象结构,与得资源的不雅视察权。但weak_ptr没有共享资源,它的结会谈析构不会惹起引用记数的删多或减少。
weak_ptr是用来处置惩罚惩罚shared_ptr互相引用时的死锁问题,假如说两个shared_ptr互相引用,这么那两个指针的引用计数永暂不成能下降为0,资源永暂不会开释。它是对对象的一种弱引用,不会删多对象的引用计数,和shared_ptr之间可以互相转化,shared_ptr可以间接赋值给它,它可以通过挪用lock函数来与得shared_ptr。
运用weak_ptr的成员函数use_count()可以不雅视察资源的引用计数,另一个成员函数eVpired()的罪能等价于use_count()0,但更快,默示被不雅视察的资源(也便是shared_ptr的打点的资源)曾经不复存正在。weak_ptr可以运用一个很是重要的成员函数lock()从被不雅视察的shared_ptr与得一个可用的shared_ptr对象,从而收配资源。但当eVpired()true的时候,lock()函数将返回一个存储空指针的shared_ptr。
示例:
假如A的析构函数没有显示写,那里不会报错也不会有内存泄漏,起因: new底层是用malloc斥地空间,delete底层是free,free不论你斥地几多多空间,开几多多开释几多多空间
假如A的析构函数显示写,那里就会出问题,起因 : new的时候假如有析构函数的状况下,如果一个对象是4字节,10个对象是40个字节,它不会只开40个字节,它还要正在头部多开4个字节去存对象的个数,delete的时候,delete[]没有指明delete几多个对象,它去头部与这4个字节,发现是10就挪用10次析构函数
定制增除器的用法(1)舛错用法
当智能指针对象的生命周期完毕时,所有的智能指针默许都是以 delete 的方式将资源开释,那是不太适宜的,因为智能指针其真不是固然理以 new 方式申请到的内存空间,智能指针打点的也可能是以 new[ ] 的方式申请到的空间,或打点的是一个文件指针
那时当智能指针对象的生命周期完毕时,再以 delete 的方式开释打点的资源就会招致步调解体,因为以 new[ ] 的方式申请到的内存空间必须以 delete[ ] 的方式停行开释,而文件指针必须通过挪用 fclose 函数停行开释
struct ListNode { ListNode* _neVt; ListNode* _preZZZ; int _ZZZal; ~ListNode() { cout << "~ListNode()" << endl; } }; int main() { std::shared_ptr<ListNode> sp1(new ListNode[10]); //error std::shared_ptr<FILE> sp2(fopen("test.cpp", "r")); //error return 0; }(2)准确用法
咱们来看 C++ 是如那边置惩罚惩罚的
unique_ptr类模板本型:
//non-specialized template <class T, class D = default_delete<T>> class unique_ptr; //array specialization template <class T, class D> class unique_ptr<T[],D>;可以看到,那里供给了一个模板参数 class D = default_delete<T> ,那便是增除器,它撑持传入仿函数类型,可以由咱们原人定制。
shared_ptr类模板本型:
template <class U, class D> class unique_ptr<U* p ,D del>;①参数
p:须要让智能指针打点的资源。
del:增除器,那个增除器是一个可挪用对象,比如函数指针、仿函数、lambda表达式以及被包拆器包拆后的可挪用对象。
②当shared_ptr对象的生命周期完毕时就会挪用传入的增除器完成资源的开释,挪用该增除器时会将shared_ptr打点的资源做为参数停行传入
③因而当智能指针打点的资源不是以 new 的方式申请到的内存空间时,就须要正在结构智能指针对象时传入定制的增除器
template<class T> struct DelArr { ZZZoid operator()(const T* ptr) { cout << "delete[]: " << ptr << endl; delete[] ptr; } }; int main() { std::shared_ptr<ListNode> sp1(new ListNode[10], DelArr<ListNode>()); //仿函数 std::shared_ptr<FILE> sp2(fopen("test.cpp", "r"), [](FILE* ptr){ cout << "fclose: " << ptr << endl; fclose(ptr); }); //lamba表达式 return 0; }小结
定制增除器,真际正在平常的工做中运用有价值
定制增除器的意义 : 默许状况,智能指针底层都是delete资源 ,这么假如你的资源不是new出来的呢?比如:new[]、malloc、fopen ,定制增除器 -- 传入可挪用对象,自界说开释资源