欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++ unique_ptr、shared_ptr、weak_ptr的區(qū)別小結(jié)

 更新時間:2025年07月10日 09:46:53   作者:Zane Joy  
本文主要介紹了C++ unique_ptr、shared_ptr、weak_ptr的區(qū)別小結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

unique_ptr

所有權(quán)轉(zhuǎn)移

unique_ptr不能被拷貝和用于賦值,因為unique_ptr刪掉了這兩個函數(shù)

但是底層源碼重載了傳右值的拷貝構(gòu)造
所以可以通過std::move來通過轉(zhuǎn)移所有權(quán)

unique_ptr<Data> p6(new Data());
//不可復(fù)制構(gòu)造和賦值復(fù)制
//unique_ptr<Data>p7 = p6; 錯誤

//p6釋放所有權(quán) 轉(zhuǎn)移到p7
unique_ptr<Data>p7 = move(p6);

unique_ptr<Data>p8(new Data());
p7 = move(p8);//重新移動賦值,原有的p6會被釋放掉
//重置空間,原空間清理
p7.reset(new Data());

所有權(quán)釋放

注意!當(dāng)unique_ptr釋放所有權(quán)以后智能指針就不會再管理這塊空間,需要自己手動釋放空間!

//釋放所有權(quán)
unique_ptr<Data>p9(new Data());
auto ptr9 = p9.release();//注意,release釋放所有權(quán)以后要自己清理空間
delete ptr9;//!?。。。?!

make_unique 和直接(new)unique_ptr

區(qū)別一 :需要一次性處理多個資源分配的地方make_unique比unique_ptr更安全

void process_data(
    std::unique_ptr<Data> p1, 
    std::unique_ptr<Data> p2
);

process_data(
    std::unique_ptr<Data>(new Data("A")),  // 分配資源 A
    std::unique_ptr<Data>(new Data("B"))   // 分配資源 B
);

編譯器在構(gòu)造函數(shù)參數(shù)時,?執(zhí)行順序是不確定的??赡艿膱?zhí)行順序例如:

  1. new Data(“A”) → 成功,得到一個裸指針 A
  2. new Data(“B”) → 成功,得到一個裸指針 B*
  3. 構(gòu)造unique_ptr 接管 A*
  4. 構(gòu)造 unique_ptr 接管 B*

如果中間發(fā)生異常:

  1. new Data(“A”) → 成功,得到 A*
  2. new Data(“B”) → ?拋出異常(例如內(nèi)存不足)? ?此時 A* 尚未被 unique_ptr 接管!? 異常被拋出后,裸指針 A* 無法被自動釋放 → ?內(nèi)存泄漏。

如果選用make_unique

process_data(
    std::make_unique<Data>("A"),  // 直接構(gòu)造并接管資源 A
    std::make_unique<Data>("B")   // 直接構(gòu)造并接管資源 B
);

此時每一步的執(zhí)行:

  1. make_unique(“A”) → ?立即構(gòu)造對象并封裝到 unique_ptr,無裸指針暴露
  2. make_unique(“B”) → 同上 如果 make_unique(“B”) 拋出異常:
    ?make_unique(“A”) 已經(jīng)返回的 unique_ptr 會正常析構(gòu) → 資源 A 被自動釋放沒有泄漏!

區(qū)別二 :直接new支持自定義刪除器

new支持在構(gòu)造 unique_ptr 時指定自定義刪除器

std::unique_ptr<MyClass, Deleter> p(new MyClass, custom_deleter);

但是make_unique不支持,只能使用默認(rèn)的 delete 操作符。

shared_ptr

shared_ptr智能指針指向同一個對象的不同成員

sc2和sc3 分別 指向sc1的index1成員和index2成員,使sc1的引用計數(shù)+2

class Data
{
public:
	Data() {

		cout<< "Begin Data" << endl;
	}
	~Data() { cout<< "End Data" << endl; }
	int index1 = 0;
	int index2 = 0;
};

{
	shared_ptr<Data>sc1(new Data);
	//打印引用計數(shù) = 1
	cout << "sc1.use_count() = " << sc1.use_count() << endl;

	shared_ptr<int>sc2(sc1,&sc1->index1);//引用計數(shù)+1
	shared_ptr<int>sc3(sc1, &sc1->index2);//引用計數(shù)+1
	//打印引用計數(shù) = 3
	cout << "sc1.use_count() = " << sc1.use_count() << endl;
}

循環(huán)引用問題

當(dāng)兩個或多個對象通過 shared_ptr ?互相持有對方時,它們的引用計數(shù)永遠(yuǎn)不會歸零,導(dǎo)致內(nèi)存無法釋放。

	class A
	{
	public:
		A() { cout << "Create A" << endl; }
		~A() { cout << " Drop A " << endl; }

		void Do()
		{
			cout << "Do b2.use_count() = " << b2.use_count() << endl;
			auto b = b2.lock(); //復(fù)制一個shared_ptr 引用計數(shù)加一
			cout << "Do b2.use_count() = " << b2.use_count() << endl;

		}

		shared_ptr<B> b1;//強智能指針
		weak_ptr<B> b2;  //弱智能指針
	};
	class B
	{
	public:
		B() { cout << "Create B" << endl; }
		~B(){ cout << " Drop B " << endl; }
		shared_ptr<A> a1;//強智能指針
		weak_ptr<A> a2;  //弱智能指針

	};

	{
		auto a = make_shared<A>();//a引用計數(shù)+1
		auto b = make_shared<B>();//b引用計數(shù)+1

		a->b1 = b;//b引用計數(shù)+1
		//出作用域前,引用計數(shù)為2
		cout << "a->b1 = b;b.use_count()=" << b.use_count() << endl;

		b->a1 = a;//a引用計數(shù)+1
		//出作用域前,引用計數(shù)為2
		cout << "b->a1 = a;a.use_count()=" << a.use_count() << endl;

	}

由于調(diào)用問題,導(dǎo)致出作用域以后a和b的引用計數(shù)都還是1,所以空間沒有被釋放

改成使用weak_ptr

	{
		auto a = make_shared<A>();
		auto b = make_shared<B>();

		a->b2 = b;//weak_ptr 引用計數(shù)不加一
		a->Do();//Do函數(shù)里引用計數(shù)加一,出Do函數(shù)作用域減一
		cout << "a->b2 = b;b.use_count()=" << b.use_count() << endl;

		b->a2 = a;//引用計數(shù)不加一
		cout << "b->a2 = a;a.use_count()=" << a.use_count() << endl;

	}//不會產(chǎn)生循環(huán)引用問題

為什么weak_ptr能解決shared_ptr的循環(huán)引用問題

weak_ptr 本身不擁有資源所有權(quán)
它只是觀察 shared_ptr 管理的對象,不會增加引用計數(shù)。

所以a->b2 = b;這句不會增加b的引用計數(shù)
同理b->a2 = a;這句也不會增加a的引用計數(shù)

但是由于weak_ptr只是一個觀察者,無法訪問任何資源,僅“觀察”資源,不擁有所有權(quán)如果想要訪問資源該怎么辦呢?

如同a->Do();里做的那樣
只要原 shared_ptr(即 a)未釋放資源,就可以通過 lock() 獲取有效的 shared_ptr 并訪問。

void Do()
{
	cout << "Do b2.use_count() = " << b2.use_count() << endl;
	auto b = b2.lock(); //返回一個shared_ptr 引用計數(shù)加一
	
	//這里還可以做其他的訪問shared_ptr資源的操作

	cout << "Do b2.use_count() = " << b2.use_count() << endl;

}//出作用域加上的那個引用計數(shù)自動-1

通過weak_ptr.lock()返回 一個 shared_ptr 并將引用計數(shù)加一

(注意!只有當(dāng)原shared_ptr對象還存在的時候才會返回shared_ptr,否者返回nullptr)

除了放函數(shù)里,還能放判斷條件里

    if (auto a_shared = b->a_weak.lock()) {
        // 返回nullptr 說明shared_ptr被釋放
        //不會執(zhí)行此處
    } else {
 		//引用計數(shù)+1
        std::cout << "A is already destroyed!" << std::endl;  // 輸出此句
    }//引用計數(shù)-1

通過這種方式,weak_ptr 可以安全地觀察資源,而不會導(dǎo)致循環(huán)引用或內(nèi)存泄漏

make_shared 和 直接 new 一個shared_ptr的區(qū)別

區(qū)別一:make_shared資源分配更安全

原因和make_unique一樣,這里就不再贅述了

區(qū)別二:直接new支持自定義刪除器

區(qū)別三:內(nèi)存分配方式不同

由于shared_ptr除了維護(hù)對象本身的內(nèi)存以外還要維護(hù)一個控制塊

  • ?強引用計數(shù)?(use_count):記錄有多少個 shared_ptr 共享對象
  • ?弱引用計數(shù)?(weak_count):記錄有多少個weak_ptr 觀察對象?
  • 自定義刪除器?(如果存在)。 ?
  • 對象指針?(指向?qū)嶋H分配的對象)。

make_shared 在底層會 ?一次性分配一塊連續(xù)內(nèi)存,既存儲對象本身,也存儲控制塊。

但是new shared_ptr會有兩次分配

std::shared_ptr<MyClass> p(new MyClass);
  1. 第一次分配:new MyClass 分配對象內(nèi)存。
  2. ?第二次分配:shared_ptr 構(gòu)造函數(shù)內(nèi)部為控制塊分配內(nèi)存。

為什么make_unique和直接new unique_ptr都是一次分配內(nèi)存

因為unique_ptr 不需要維護(hù)引用計數(shù),因此 ?沒有控制塊。無論通過 make_unique 還是直接 new,都只需分配對象內(nèi)存

shared_ptr合并分配的缺點

對象和控制塊內(nèi)存綁定,即使所有 shared_ptr 銷毀,若仍有 weak_ptr 存在,對象內(nèi)存仍然需等待控制塊釋放(但析構(gòu)函數(shù)會被及時調(diào)用)

shared_ptr控制塊內(nèi)存的延遲釋放是內(nèi)存泄漏嗎?

由于make_shared 是一次性分配一塊連續(xù)內(nèi)存,同時存儲 ?對象實例 和 ?控制塊?(包含引用計數(shù)、弱引用計數(shù)等)

所以當(dāng)沒有weak_ptr存在時

通過make_shared建立的shared_ptr:

  • 對象析構(gòu)函數(shù)立即被調(diào)用。
  • 整塊內(nèi)存(對象 + 控制塊)立即釋放。

通過new建立的shared_ptr:

  • 對象內(nèi)存立即釋放。
  • 控制塊內(nèi)存也立即釋放。

當(dāng)有weak_ptr存在時

通過make_shared建立的shared_ptr:

  • 對象析構(gòu)函數(shù)被調(diào)用(資源清理)。 ?
  • 對象內(nèi)存和控制塊內(nèi)存暫時保留?(直到所有 weak_ptr 也被銷毀)。

通過new建立的shared_ptr:

  • 對象內(nèi)存立即釋放(僅保留控制塊內(nèi)存)。

由于make_shared對象和控制塊內(nèi)存是連續(xù)的,無法單獨釋放對象內(nèi)存。所以必須等待所有 weak_ptr 銷毀后,?整塊內(nèi)存才能一起釋放。

那這是內(nèi)存泄漏嗎?

不是!? 內(nèi)存泄漏的定義是:?無法再訪問且未釋放的內(nèi)存。
而在此時:
對象析構(gòu)函數(shù)已被調(diào)用(資源已清理)。
內(nèi)存仍被 weak_ptr 的控制塊管理,雖然未釋放,但程序仍能通過 weak_ptr 的機制感知到內(nèi)存狀態(tài)。
當(dāng)所有 weak_ptr 銷毀后,內(nèi)存會被正確釋放。

enable_shared_from_this 和 shared_from_this()

解決場景:

當(dāng)一個對象已被 shared_ptr 管理時,若在成員函數(shù)中直接用 this 創(chuàng)建新的 shared_ptr,會導(dǎo)致 多個獨立的引用計數(shù) 。當(dāng)某一 shared_ptr 析構(gòu)時,對象可能被提前銷毀,引發(fā)懸空指針或雙重釋放。

問題場景示例1

class BackgroundWorker {
public:
    void startWork() {
        // 錯誤:裸指針 this 跨線程傳遞,主線程可能提前釋放對象
        std::thread([this]() {
            doWork(); // 可能訪問已銷毀對象!
        }).detach();
    }

    void doWork() { /* 自己實現(xiàn)的操作 */ }
};

int main() {
    auto worker = std::make_shared<BackgroundWorker>();
    worker->startWork();
    worker.reset(); // 主線程釋放對象
    // 子線程仍在執(zhí)行 doWork(),this 已失效!
}

enable_shared_from_this 內(nèi)部保存一個 weak_ptr,指向與對象關(guān)聯(lián)的 shared_ptr 控制塊。通過 shared_from_this() 方法生成與已有 shared_ptr 共享所有權(quán)的 shared_ptr 確保引用計數(shù)一致。

解決示例
通過獲取與 shared_ptr 共享所有權(quán)的智能指針,延長對象生命周期

class BackgroundWorker : 
public std::enable_shared_from_this<BackgroundWorker> {
public:
    void startWork() {
        // 正確:通過 shared_from_this() 傳遞所有權(quán)
        std::thread([self = shared_from_this()]() {
            self->doWork(); // 安全:self 確保對象存活
        }).detach();
    }
    // ...
};

問題場景示例2:

class TreeNode {
public:
    void addChild() {
        // 創(chuàng)建一個子節(jié)點 child,用 shared_ptr 管理
        auto child = std::make_shared<TreeNode>();
        
        // 錯誤:直接用 this 創(chuàng)建新的 shared_ptr,賦值給 child->parent
        //child->parent引用計數(shù)也是 1
        child->parent = std::shared_ptr<TreeNode>(this); // 危險!
        
        // 將 child 添加到當(dāng)前節(jié)點的 children 列表中
        children.push_back(child);
    }

    std::shared_ptr<TreeNode> parent;  // 父節(jié)點用 shared_ptr 管理
    std::vector<std::shared_ptr<TreeNode>> children; // 子節(jié)點列表
};

int main() {
    // 創(chuàng)建一個根節(jié)點 root,用 shared_ptr 管理
    auto root = std::make_shared<TreeNode>();//root引用計數(shù)是 1
    root->addChild(); // 調(diào)用 addChild,導(dǎo)致雙重釋放!
}

這段代碼的關(guān)鍵問題出現(xiàn)在 child->parent = std::shared_ptr(this)
this 是當(dāng)前 TreeNode 對象的裸指針(即 root 指向的對象)。
直接用 this 創(chuàng)建了一個新的 shared_ptr,賦值給 child->parent。這個新的 shared_ptr 與 root 是 ?完全獨立 的,它不知道 root 的存在,因此它的引用計數(shù) “也是 ?1”。

當(dāng) main 函數(shù)結(jié)束時,root 的引用計數(shù)減為 ?0,會釋放它指向的 TreeNode 對象。

但此時 child->parent 也指向同一個 TreeNode 對象,且它的引用計數(shù)仍然是 ?1?(如果 child 未被銷毀)。當(dāng) child 析構(gòu)時,child->parent 的引用計數(shù)減為 ?0,會再次嘗試釋放同一個 TreeNode 對象,導(dǎo)致** ?雙重釋放**。

解決示例:

class TreeNode : public std::enable_shared_from_this<TreeNode> {
public:
    void addChild() {
        auto child = std::make_shared<TreeNode>();
        
        // 通過 shared_from_this() 獲取與 root 共享所有權(quán)的 shared_ptr
        child->parent = shared_from_this(); // 引用計數(shù) +1
        
        children.push_back(child);
    }

    std::shared_ptr<TreeNode> parent;
    std::vector<std::shared_ptr<TreeNode>> children;
};

int main() {
    auto root = std::make_shared<TreeNode>();
    root->addChild(); // 安全:root 的引用計數(shù)變?yōu)?2
}

shared_ptr的線程安全問題

http://www.dbjr.com.cn/article/41353.htm

到此這篇關(guān)于C++ unique_ptr、shared_ptr、weak_ptr的區(qū)別小結(jié)的文章就介紹到這了,更多相關(guān)C++ unique_ptr shared_ptr weak_ptr內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用C++ Matlab中的lp2lp函數(shù)教程詳解

    使用C++ Matlab中的lp2lp函數(shù)教程詳解

    本文介紹如何使用C++編寫數(shù)字濾波器設(shè)計算法,實現(xiàn)Matlab中的lp2lp函數(shù),將低通濾波器轉(zhuǎn)換為參數(shù)化的低通濾波器,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2023-04-04
  • C語言實現(xiàn)修改文本文件中特定行的實現(xiàn)代碼

    C語言實現(xiàn)修改文本文件中特定行的實現(xiàn)代碼

    最近由于項目需要實現(xiàn)修改文件的功能,所以,博主認(rèn)真查閱了一些資料,但是,很遺憾,并沒有太多的收獲
    2013-06-06
  • C++實現(xiàn)模板中的非類型參數(shù)的方法

    C++實現(xiàn)模板中的非類型參數(shù)的方法

    這篇文章主要介紹了C++實現(xiàn)模板中的非類型參數(shù)的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • Cocos2d-x 3.x入門教程(一):基礎(chǔ)概念

    Cocos2d-x 3.x入門教程(一):基礎(chǔ)概念

    這篇文章主要介紹了Cocos2d-x 3.x入門教程(一):基礎(chǔ)概念,本文講解了Director、Scene、Layer、Sprite等內(nèi)容,需要的朋友可以參考下
    2014-11-11
  • vc6.0中c語言控制臺程序中的定時技術(shù)(定時器)

    vc6.0中c語言控制臺程序中的定時技術(shù)(定時器)

    這篇文章主要介紹了vc6.0中c語言控制臺程序中的定時技術(shù)(定時器),需要的朋友可以參考下
    2014-04-04
  • C語言菜鳥基礎(chǔ)教程之加法

    C語言菜鳥基礎(chǔ)教程之加法

    C語言中運算符和表達(dá)式數(shù)量之多, 在高級語言中是少見的。正是豐富的運算符和表達(dá)式使C語言功能十分完善。 這也是C語言的主要特點之一。今天我們來看看加法運算
    2017-10-10
  • C++實現(xiàn)二叉樹基本操作詳解

    C++實現(xiàn)二叉樹基本操作詳解

    這篇文章主要為大家詳細(xì)介紹了C++實現(xiàn)二叉樹基本操作,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • 算法詳解之回溯法具體實現(xiàn)

    算法詳解之回溯法具體實現(xiàn)

    這篇文章主要介紹了算法詳解之回溯法具體實現(xiàn),需要的朋友可以參考下
    2014-02-02
  • C語言經(jīng)典例程100例(經(jīng)典c程序100例)

    C語言經(jīng)典例程100例(經(jīng)典c程序100例)

    這篇文章主要介紹了C語言經(jīng)典例程100例,經(jīng)典c程序100例,學(xué)習(xí)c語言的朋友可以參考一下
    2018-03-03
  • 使用C++構(gòu)建一個優(yōu)先級隊列的實現(xiàn)

    使用C++構(gòu)建一個優(yōu)先級隊列的實現(xiàn)

    優(yōu)先級隊列是一種特殊的隊列數(shù)據(jù)結(jié)構(gòu),本文主要介紹了使用C++構(gòu)建一個優(yōu)先級隊列的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2025-02-02

最新評論