聊聊 C++ 中的幾種智能指針 (下)

来源:https://www.cnblogs.com/huangxincheng/archive/2022/07/24/16514090.html
-Advertisement-
Play Games

一:背景 上一篇我們聊到了C++ 的 auto_ptr ,有朋友說已經在 C++ 17 中被棄用了,感謝朋友提醒,今天我們來聊一下 C++ 11 中引入的幾個智能指針。 unique_ptr shared_ptr weak_ptr 看看它們都怎麼玩。 二:三大智能指針詳解 1. unique_ptr ...


一:背景

上一篇我們聊到了C++ 的 auto_ptr ,有朋友說已經在 C++ 17 中被棄用了,感謝朋友提醒,今天我們來聊一下 C++ 11 中引入的幾個智能指針。

  1. unique_ptr

  2. shared_ptr

  3. weak_ptr

看看它們都怎麼玩。

二:三大智能指針詳解

1. unique_ptr

上一篇在說 auto_ptr 的時候聊過一個術語叫 控制權轉移,言外之意就是下麵的代碼會 訪問違例 。


void test() {
	int* i = new int(10);

	auto_ptr<int> ptr1(i);
	auto_ptr<int> ptr2 = ptr1;  //將 ptr1 的控制台送給了 ptr2

	cout << *ptr1 << endl;
}

int main() {
	test();
}

這段代碼用了賦值運算符,可能是程式員的無心之舉,卻引來程式的崩潰,那能不能規避下這種 無心 的行為呢? 當然是可以的,屏蔽掉 auto_ptr 中的賦值運算符不就可以了哈,C++ 中的 unique_ptr 就是這麼實現的,可以看下源碼。


class unique_ptr {

public:
	unique_ptr(const unique_ptr&) = delete;
	unique_ptr& operator=(const unique_ptr&) = delete;
};

可以看到,unique_ptr 將 =,複製構造函數 都給屏蔽了,所以再硬寫的話,必然是報錯。

2. shared_ptr

這個算是純正的 引用計數,高級語言中的 引用跟蹤 經常會拿它做比較,說它存在迴圈引用等各種問題。

在迴圈引用之前,先說下它的簡單用法。


void test() {
	int* i = new int(10);
	shared_ptr<int> ptr1(i);
	shared_ptr<int> ptr2 = ptr1;
	shared_ptr<int> ptr3 = ptr2;

	printf("ptr.refcnt= %d, ptr2.refcnt=%d, ptr3.refcnt=%d \n", 
												   ptr1.use_count(), 
												   ptr2.use_count(), 
												   ptr3.use_count());
}

int main() {
	test();
}

再看下源碼,大概在內部共同維護了對 _Ref_count_base 引用。


class _Ref_count_base {
private:
	_Atomic_counter_t _Uses = 1;
	_Atomic_counter_t _Weaks = 1;
}

很顯然當所有的析構函數將 _Uses 降為 0 的時候就會對 i 進行釋放,參考源碼如下:


class shared_ptr : public _Ptr_base<_Ty> {
	~shared_ptr() noexcept { // release resource
		this->_Decref();
	}
}

class _Ref_count_base {

	void _Decref() noexcept { // decrement use count
		if (_MT_DECR(_Uses) == 0) {
			_Destroy();
			_Decwref();
		}
	}
}

原理大概摸清楚了,接下來看下迴圈引用,它最大的問題就是讓 _Uses 永遠也不會為 0,


#include <iostream>
#include <string>

using namespace std;

class BClass;

class AClass
{
public:
	int i = 10;
	shared_ptr<BClass> b;
};


class BClass
{
public:
	int i = 11;
	shared_ptr<AClass> a;
};

void test() {

	AClass* a = new AClass();
	BClass* b = new BClass();

	shared_ptr<AClass> aclass(a);
	shared_ptr<BClass> bclass(b);

	aclass->b = bclass;
	bclass->a = aclass;

	printf("a.refcnt=%d, b.refcnt=%d, a=%x,b=%x \n", aclass.use_count(), bclass.use_count(), &(*a), &(*b));
}

int main() {
	test();
}

接下來可以在 main 方法結束的地方觀察 a,b 所有的 heap 塊的記憶體內容是否被釋放? 可以清楚的看到還在那裡。。. 如下圖所示:

這就是 迴圈引用 造成的問題,這時候就需要用到 weak_ptr 了。

3. weak_ptr

弱引用就是它的引用不改變對原對象中 _Uses 的引用,接下來將 AClass 和 BClass 中的 shared_ptr 改成 weak_ptr 版。


class AClass
{
public:
	int i = 10;
	weak_ptr<BClass> b;
};


class BClass
{
public:
	int i = 11;
	weak_ptr<AClass> a;
};

從圖中可以看到,此時 refcnt=1, 再觀察下地址 0x007f71a8 的內容,可以發現已經被釋放啦。

好了,今天我們就聊這麼多,希望對大家有幫助。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • PHP變數規則 變數以$符號開頭,其後是變數的名稱; 變數名稱必須已字母或者下劃線開頭; 變數名稱不能以數字開頭; 變數名稱只能包含字母數字字元和下劃線(A-z、0-9以及 _ ); 變數名稱對大小寫敏感($y和$Y是兩個不同的變數) 註釋:PHP變數名稱對大小寫敏感 創建PHP變數 PHP變數沒有 ...
  • java變數類型 1、局部變數 概念:類的方法中的變數 2、實例變數 概念:定義在類裡面,方法體、構造方法和語句塊外,且不被static修飾的變數,用於描述一個類中包含哪些數據 3、類變數 概念:獨立於方法之外的變數,有static修飾 public class able{ static int a ...
  • 基於SSM框架開發的CRM客戶管理系統,適合剛學完SSM的同學,幫助夯實javase到ssm之間的知識,提升學生的邏輯思維,也瞭解到企業軟體開發的流程及代碼編寫規範。 ...
  • 看《C++ Primer Plus》時整理的學習筆記,部分內容完全摘抄自《C++ Primer Plus》(第6版)中文版,Stephen Prata 著,張海龍 袁國忠譯,人民郵電出版社。只做學習記錄用途。 ...
  • 本文是在嘗試了刷新Maven項目、clean了Maven緩存並且重啟IDEA之後任然無法在Tomcat中找到子模塊對應的工件,這時就要試著模仿著自己創建一個模塊 父類的pom.xml模塊的配置,以及子模塊pom.xml的parent中寫的父模塊,這些都寫好了,在Tomcat中任然無法找到該模塊對應的 ...
  • 1 數組 1.1 數組概念 簡單來說,普通變數只能存取單個數據元素,與之不同,數組可以存取多個數據元素,記憶體中數組以順序結構存儲。 定義1:(數組) 同一類型數組中存取的元素類型相同,並且各個元素之間連續存儲。 1.2 整形數組 1.2.1 一維數組 (1)定義方式 類型說明符 數組名[常量表達式] ...
  • 多態 多態的定義 多態按字面的意思就是多種形態。當類之間存在層次結構,並且類之間是通過繼承關聯時,就會用到多態。C++ 多態意味著調用成員函數時,會根據調用函數的對象的類型來執行不同的函數。 簡單來說就是完成某個行為,不同的對象會產生不同的結果 多態的構成條件 1. 必須通過基類的指針或者引用調用虛 ...
  • 介紹 這是我隨手寫的一個小程式,希望大家能從中學習到 列表 與 open() 函數,感受Python的魅力! #代碼瀏覽 點擊查看代碼 #獲取圖書 #從文件中讀取圖書,並寫入列表 def getBook(): bookList =[] f = open("book.txt","r") tempLis ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...