亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C++11中的智能指針和垃圾回收使用

 更新時間:2023年02月02日 10:01:06   作者:Shawn-Summer  
本文主要介紹了C++11中的智能指針和垃圾回收使用,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

在C/C++中,我們需要自己管理動態(tài)內(nèi)存區(qū),我們在寫代碼中可能會出現(xiàn)如下3中內(nèi)存管理的缺陷

  • 野指針:內(nèi)存單元已經(jīng)釋放,但是指向它的指針還在使用
  • 重復(fù)釋放:試圖是釋放已經(jīng)釋放過的內(nèi)存單元
  • 內(nèi)存泄漏:不再使用的內(nèi)存單元沒有進行釋放

C++惡心的地方就在于它存在指針,需要編寫者自己管理內(nèi)存,所以內(nèi)存上面的問題就會有很多,但是在其他語言,例如python,java,C#,他都不存在指針概念,也就意味著你不需要開辟釋放內(nèi)存這些操作。而正因為C++將指針暴露出來,甚至將右值引用暴露出來,才使得C++的運行效率非???。

為了減少C++中的內(nèi)存問題,就出現(xiàn)了智能指針,它是一種對C風(fēng)格指針的優(yōu)化,它把內(nèi)存的釋放放在了智能指針的析構(gòu)函數(shù)中,這樣子就能減少一部分自己手動釋放內(nèi)存的代碼。

1.C++11中的unique_ptr

#include<memory>
#include<iostream>
using namespace std;
int main()
{
? ? unique_ptr<int> up1(new int(11));
? ? unique_ptr<int> up2=up1;//無法通過編譯

? ? cout<<*up1<<endl;//11
? ? unique_ptr<int> up3=move(up1);//現(xiàn)在up3是數(shù)據(jù)的唯一指針

? ? cout<<*up3<<endl;//11
? ? cout<<*up1<<endl;//運行錯誤

? ? up3.reset();//釋放內(nèi)存
? ? up1.reset();//不會重復(fù)釋放內(nèi)存

? ? cout<<*up3<<endl;//運行錯誤
}

我們知道unique_ptr正如它的名字一樣,它表示一個對象只能由一個指針綁定,不允許一個對象同時又多個unique_ptr綁定。

而且 unique_ptr只存在移動語義,而不存在拷貝語義 ,我們看上面代碼中unique_ptr<int> up3=move(up1);,在unique_ptr中只存在移動構(gòu)造函數(shù)和移動賦值函數(shù),不存在拷貝構(gòu)造函數(shù)和拷貝賦值函數(shù)。所以說我們只能用右值來構(gòu)造或賦值unique_ptr。

還有一種初始化unique_ptr的方法就是:make_unique<>(),相較于使用new初始化,前者內(nèi)存碎片化更少,在現(xiàn)代C++種主要使用,make_unique。

實際上,C++98中的auto_ptr和C++11中的unique_ptr實現(xiàn)的是同一個東西,但是在C++98中我們不存在移動語義,所以auto_ptr它是存在拷貝構(gòu)造函數(shù)和拷貝賦值函數(shù)的,所以諸如:
auto_ptr<int> up2=up1;是可以通過編譯的,在C++11中我們廢棄掉了,auto_ptr也是這個原因。

2.C++11中的shared_ptr和weak_ptr

shared_ptr是一種共享式的指針,它采用引用計數(shù)的方式,來決定何時釋放內(nèi)存,引用計數(shù)就是說,它統(tǒng)計每個對象有幾個指針指向它。一旦一個對象的引用計數(shù)為0,即不存在指向它的指針,那么就釋放它。
weak_ptr是用來驗證shared_ptr指向的內(nèi)存單元的有效性的,被它指向的對象的引用計數(shù)不會增加。

#include<memory>
#include<iostream>
using namespace std;

void Check(weak_ptr<int>& wp)
{
? ? shared_ptr<int> sp=wp.lock();
? ? if(sp!=nullptr)
? ? ? ? cout<<"still "<<*sp<<endl;
? ? else
? ? ? ? cout<<"pointer is invalid."<<endl;
}
int main()
{
? ? shared_ptr<int> sp1=make_shared<int>(22);
? ? shared_ptr<int> sp2=sp1;
? ? weak_ptr<int> wp=sp1;

? ? cout<<*sp1<<endl;
? ? cout<<*sp2<<endl;
? ? Check(wp);

? ? sp1.reset();
? ? cout<<*sp2<<endl;
? ? Check(wp);

? ? sp2.reset();
? ? Check(wp);

}

22
22
still 22
22
still 22
pointer is invalid.

3.垃圾回收

雖然智能指針能夠幫助用戶有效管理堆內(nèi)存,但是它還是需要顯式聲明智能指針,而完全不需要指針的內(nèi)存管理方法也會更討人喜歡。這種方法就是垃圾回收機制,寫代碼的時候不需要開辟釋放內(nèi)存操作,這些操作都由編譯器自動實現(xiàn),這種智能化的方案就是垃圾回收機制。

遺憾的是,C++不支持垃圾回收機制。

垃圾回收的方式有4種

基于引用計數(shù)的方法
其實就是和shared_ptr一樣的方式,就是一旦對象的引用次數(shù)為0就釋放它,python就是使用的這種方案,不過這種方案不好,它效率比較低,一旦對象創(chuàng)建,或者有指針指向它,都要計算引用此時,而且它不能解決"環(huán)形引用"問題

標(biāo)記-清除
這種方法就是存在一個根對象,它管理所有對象,依次遍歷每個對象,給它們引用的區(qū)域打上標(biāo)記,然后遍歷完成后,把所有沒有標(biāo)記的區(qū)域釋放掉,這種方案的缺陷在于會存在大量的內(nèi)存碎片

標(biāo)記-整理
它是在標(biāo)記-清除方案的基礎(chǔ)上,標(biāo)記完后不再遍歷釋放垃圾了,而是所有被標(biāo)記的區(qū)域,向左靠齊,這樣就減少了內(nèi)存碎片

標(biāo)記-拷貝
它是將內(nèi)存空間分為兩塊:From和to,剛開始就從From空間種分配內(nèi)存,一旦From內(nèi)存滿了,就把From空間中所有活對象,拷貝到to空間中,而且都是向左靠齊的,然后再將From和to的角色互換。

很遺憾C++11目前還沒有公開支持過垃圾回收,不過有些庫和有限編譯器支持了部分垃圾回收的功能

int main()
{
? ? int *p=new int;
? ? p+=10;
? ? p-=10;
? ? *p=10;
}

上面代碼中,一旦p指向了其他區(qū)域,如果你的編譯器支持垃圾回收,例如采用的引用計時方式,那么一旦p移到了其他地區(qū),這個開辟的new int空間,就會被釋放,更危險的是,這塊空間會被其他線程使用,這時候,p如果又指回了原來的地方,那么p就是一個野指針。

為了防止,new int這塊空間被垃圾回收器回收掉,我們的一種方案是:

int main()
{
    int *p=new int;
    declare_reachable(p);
    p+=10;
    p-=10;
    *p=10;
}

這里的declare_reachable函數(shù)顯式的告訴垃圾回收器,你不要取釋放這塊空間

到此這篇關(guān)于C++11中的智能指針和垃圾回收使用的文章就介紹到這了,更多相關(guān)C++11 智能指針和垃圾回收內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言數(shù)據(jù)結(jié)構(gòu)實現(xiàn)銀行模擬

    C語言數(shù)據(jù)結(jié)構(gòu)實現(xiàn)銀行模擬

    這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)實現(xiàn)銀行模擬的相關(guān)資料,通過此文希望大家能理解離散化的方法,希望能幫助到大家,需要的朋友可以參考下
    2017-08-08
  • C語言實現(xiàn)一個簡單的掃雷游戲

    C語言實現(xiàn)一個簡單的掃雷游戲

    掃雷是電腦上很經(jīng)典的游戲,特意去網(wǎng)上玩了一會,幾次調(diào)試之后,發(fā)現(xiàn)這個比三子棋要復(fù)雜一些,尤其是空白展開算法上和堵截玩家有的一拼,與實際游戲差別較大,不能使用光標(biāo),下面來詳解每一步分析
    2021-10-10
  • 基于C語言實現(xiàn)泛型編程詳解

    基于C語言實現(xiàn)泛型編程詳解

    對于C而言,想實現(xiàn)泛型編程并非易事,甚至可以說非常繁瑣,一大堆坑。最主要也沒有現(xiàn)成的輪子可用。本文就來詳細為大家講講C語言如何實現(xiàn)泛型編程詳解,需要的可以參考一下
    2022-07-07
  • Reactor反應(yīng)器的實現(xiàn)方法詳解

    Reactor反應(yīng)器的實現(xiàn)方法詳解

    本篇文章是對Reactor反應(yīng)器的實現(xiàn)方法進行了詳細的分析介紹,需要的朋友參考下
    2013-05-05
  • Opencv學(xué)習(xí)教程之漫水填充算法實例詳解

    Opencv學(xué)習(xí)教程之漫水填充算法實例詳解

    這篇文章主要給大家介紹了Opencv學(xué)習(xí)教程之漫水填充算法的相關(guān)資料,文中給出了詳細的示例代碼供大家參考學(xué)習(xí),對大家具有一定的參考價值,需要的朋友們下面跟著小編一起來學(xué)習(xí)學(xué)習(xí)吧。
    2017-06-06
  • C++中vector類的一些簡單實現(xiàn)

    C++中vector類的一些簡單實現(xiàn)

    C++中的std::vector是一個動態(tài)數(shù)組(也被稱為可變大小數(shù)組)的容器類,它是C++標(biāo)準(zhǔn)庫提供的其中一種容器類,提供了方便的操作和管理動態(tài)數(shù)組的功能,本文就給大家介紹了C++中vector類的簡單實現(xiàn)代碼,需要的朋友可以參考下
    2023-08-08
  • C語言實現(xiàn)簡單的推箱子小游戲

    C語言實現(xiàn)簡單的推箱子小游戲

    這篇文章主要為大家詳細介紹了C語言實現(xiàn)簡單的推箱子小游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-07-07
  • 項目之C++如何實現(xiàn)數(shù)據(jù)庫連接池

    項目之C++如何實現(xiàn)數(shù)據(jù)庫連接池

    這篇文章主要介紹了項目之C++如何實現(xiàn)數(shù)據(jù)庫連接池問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • 標(biāo)準(zhǔn)CSV格式的介紹和分析以及解析算法實例詳解

    標(biāo)準(zhǔn)CSV格式的介紹和分析以及解析算法實例詳解

    這篇文章主要介紹了標(biāo)準(zhǔn)CSV格式的介紹和分析以及解析算法實例詳解的相關(guān)資料,需要的朋友可以參考下
    2016-12-12
  • C++ 動態(tài)數(shù)組模版類Vector實例詳解

    C++ 動態(tài)數(shù)組模版類Vector實例詳解

    這篇文章主要為大家詳細介紹了C++動態(tài)數(shù)組模版類Vector實例,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-02-02

最新評論