<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
在C++中,動態記憶體的使用時有一定的風險的,因為它沒有垃圾回收機制,很容易導致忘記釋放記憶體的問題,具體體現在異常的處理上。想要釋放掉拋異常的程式的一些記憶體,往往需要多次拋異常,這種處理方式是十分麻煩的。
智慧指標的本質就是使用一個物件來接管一段開闢的空間,在該物件在銷燬的時候,自動呼叫解構函式來釋放這段記憶體。
因此智慧指標的本質是一個類,類中最主要的物件是一個指標,該類的解構函式就是銷燬該指標指向的空間,使用智慧指標的本質就是將一個指向動態開闢空間的指標賦給該類中的指標。不過這樣的處理過程會有一定的問題,比如淺拷貝等。
C++標準庫提供了兩種智慧指標型別來管理動態物件,由於該物件的行為酷似指標,所以稱為智慧指標。它們分別是shared_ptr以及unique_ptr。還提供了一個weak_ptr它主要是為了解決shared_ptr的迴圈參照問題。
shared_ptr允許多個指標指向同一個物件,unique_ptr則獨佔所指向的物件。
在很早以前,大佬們就已經認識到了記憶體釋放的問題,因此為標準庫中增加了一個類:auto_str。它有著和unique_str智慧指標類似的功能,它雖然成功的將一個開闢的資源塞給了一個類,不過存在很嚴重的問題,一些公司已經明令禁止使用它了:
auto_ptr<int> sptr1(new int); auto_ptr<int> sptr2(sptr1); *sptr1;
此時如果對sptr1進行解除參照操作,會發生報錯。要了解報錯的原因,我們需要了解它的大致底層原理,作為第一個出現的智慧指標,它只是簡單執行了將資源轉移,以及在解構中加入資源釋放,還有一些解除參照的運運算元過載函數:
template<class T> class MyAuto { private: T* _ptr; public: MyAuto(T* ptr) :_ptr(ptr) {} ~MyAuto() { if (_ptr != nullptr) { cout << "delete: " << _ptr << endl; delete _ptr; _ptr = nullptr; } } MyAuto(MyAuto<T>& Ptr) { _ptr = Ptr._ptr; Ptr._ptr = nullptr; } T& operator*() { return *_ptr; } T* operator->() { return _ptr; } };
可以發現,最終是淺拷貝的鍋。因為在進行資源轉移的時候,必須將原來的指標置為nullptr,否則解構的時候會解構兩次。而將其置為nullptr之後再要使用該指標對其進行解除參照就會發生崩潰。
unique_ptr處理上述問題簡單而粗暴,即不讓進行拷貝操作:
unique_ptr<int> sptr1(new int); unique_ptr<int> sptr2(sptr1);
直接進行報錯處理。
我們也可以猜測出它的實現方式,那就是在拷貝構造和賦值構造的後面加上delete關鍵字。
template<class T> class MyUnique { private: T* _ptr; public: MyUnique(T* ptr) :_ptr(ptr) {} ~MyUnique() { if (_ptr != nullptr) { cout << "delete: " << _ptr << endl; delete _ptr; _ptr = nullptr; } } MyUnique(MyUnique<T>& Ptr) = delete; MyUnique& operator=(MyUnique<T>& Ptr) = delete; T& operator*() { return *_ptr; } T* operator->() { return _ptr; } };
(1)參照計數器
shared_ptr是使用最多的智慧指標,即它可以進行拷貝構造。
shared_ptr<int> sptr1(new int(1)); shared_ptr<int> sptr2(sptr1); shared_ptr<int> sptr3(sptr2); cout << sptr1.use_count() << endl; cout << sptr2.use_count() << endl; cout << sptr2.use_count() << endl; cout << "資源釋放成功" << endl;
(2)執行緒安全
涉及到共用,我們不得不將執行緒安全問題考慮進來,很顯然shared_ptr無論是要管理的資源的使用,還是要指向的該資源對應的計數器的加減操作,都不是執行緒安全的。
template<class T> class MyShared { private: T* _ptr; mutex* _pmtx; int* _pcount; public: MyShared(T* ptr) :_ptr(ptr), _pmtx(new mutex), _pcount(new int(1)) {} void AddCount() { _pmtx->lock(); (*_pcount)++; _pmtx->unlock(); } void DelCount() { _pmtx->lock(); bool flag = false; if (--(*_pcount) == 0) { if (_ptr != nullptr) { cout << "delete: " << _ptr << endl; delete _ptr; _ptr = nullptr; } delete _pcount;//當為0的時候刪除計數器 _pcount = nullptr; flag = true; } _pmtx->unlock(); if (flag == true) { delete _pmtx; _pmtx = nullptr; } } MyShared(MyShared<T>& sp) :_ptr(sp._ptr), _pcount(sp._pcount), _pmtx(sp._pmtx) { AddCount(); } MyShared& operator=(MyShared<T>& sp) { if (_ptr != sp._ptr) { DelCount();//釋放管理的舊資源 _ptr = sp._ptr; _pcount = sp._pcount; _pmtx = sp._pmtx; AddCount();//對管理的新資源的計數器進行++ } return *this; } //獲取參照計數 int use_count() { return *_pcount; } T& operator*() { return *_ptr; } T* operator->() { return _ptr; } };
(3)刪除器
如果不是new出來的物件如何通過智慧指標進行管理呢?其實shared_ptr設計了一個刪除器來解決這一問題。
template<class T> struct FreeFunc { void operator()(T* ptr) { cout << "free:" << ptr << endl; free(ptr); } }; template<class T> struct DeleteArrayFunc { void operator()(T* ptr) { cout << "delete[]" << ptr << endl; delete[] ptr; } };
此時使用malloc進行初始化的時候就也可以進行清理空間了:
FreeFunc<int> freeFunc; shared_ptr<int> sp1((int*)malloc(4), freeFunc); DeleteArrayFunc<int> deleteArrayFunc; shared_ptr<int> sp2((int*)malloc(4), deleteArrayFunc);
(1)shared_ptr中的迴圈呼叫問題
迴圈呼叫問題在一些特殊的情況下會產生:
1.node1和node2兩個智慧指標指向兩個節點,參照計數變成1,我們不需要手動delete。
2.node1的_next指向node2,node2的_prev指向node1,參照計數變成2。
3.node1和node2解構,參照計數減到1,但是_next還指向下一個節點。但是_prev還指向上一個節點。
4.也就是說_next解構了,node2就釋放了。
5.也就是說_prev解構了,node1就釋放了。
6.但是_next屬於node的成員,node1釋放了,_next才會解構,而node1由_prev管理,_prev屬於node2成員,所
以這就叫回圈參照,誰也不會釋放。
struct ListNode { shared_ptr<ListNode> _next; shared_ptr<ListNode> _prev; }; shared_ptr<ListNode> node1(new ListNode); shared_ptr<ListNode> node2(new ListNode); node1 ->_next = node2; node2 -> _prev = node1;
通俗來講,就是此時如果想釋放node2,那麼就需要delete(n1->next),但是如果要釋放n1->next就必須delete(n1),而要deleten1又需要delete(node2->prev)因此如果不讓prev指向n就沒有問題。
(2)weak_ptr
struct ListNode { std::weak_ptr<ListNode> _next; std::weak_ptr<ListNode> _prev; int _val; ~ListNode() { cout << "~ListNode()" << endl; } }; int main() { std::shared_ptr<ListNode> node1(new ListNode); std::shared_ptr<ListNode> node2(new ListNode); cout << node1.use_count() << endl; cout << node2.use_count() << endl; node1->_next = node2; node2->_prev = node1; //... cout << node1.use_count() << endl; cout << node2.use_count() << endl; return 0; }
以上就是深入瞭解C++智慧指標的使用的詳細內容,更多關於C++智慧指標的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45