在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C++中的資源泄露問題

Linux愛好者 ? 來源:一個程序員的修煉之路 ? 作者:河邊一枝柳 ? 2021-09-30 17:03 ? 次閱讀

在Modern C++之前,C++無疑是個更容易寫出坑的語言,無論從開發效率,和易坑性,讓很多新手望而卻步。比如內存泄露問題,就是經常會被寫出來的坑,本文就讓我們一起來看看,這些讓現在或者曾經的C++程序員淚流滿面的內存泄露場景吧。你是否有踩過?

1. 函數內或者類成員內存未釋放

這類問題可以稱之為out of scope的時候,并沒有釋放相應對象的堆上內存。有時候最簡單的場景,反而是最容易犯錯的。這個我想主要是因為經常寫,哪有不出錯。下面場景一看就知道了,當你在寫XXX_Class * pObj = new XXX_Class();這一行的時候,腦子里面還在默念記得要釋放pObj ,記得要釋放pObj, 可能因為重要的事情要說三遍,而你只喊了兩遍,最終還是忘記了寫delete pObj; 這樣去釋放對象。

void MemoryLeakFunction()

{

XXX_Class * pObj = new XXX_Class();

pObj-》DoSomething();

return;

}

下面這個場景,就是析構函數中并沒有釋放成員所指向的內存。這個我們就要注意了,一般當你構建一個類的時候,寫析構函數一定要切記釋放類成員關聯的資源。

class MemoryLeakClass

{

public

MemoryLeakClass()

{

m_pObj = new XXX_ResourceClass;

}

void DoSomething()

{

m_pObj-》DoSomething();

}

~MemoryLeakClass()

{

;

}

private:

XXX_ResourceClass* m_pObj;

};

上述這兩種代碼例子,是不是讓一個C++工程師如履薄冰,完全看自己的大腦在不在狀態。在boost或者C++ 11后,通過智能指針去進行包裹這個原始指針,這是一種RAII的思想(可以參閱本文末尾的關聯閱讀), 在out of scope的時候,釋放自己所包裹的原始指針指向的資源。將上述例子用unique_ptr改寫一下。

void MemoryLeakFunction()

{

std::unique_ptr《XXX_Class》 pObj = make_unique《XXX_Class》();

pObj-》DoSomething();

return;

}

2. delete []

大家知道C++中這樣一個語句XXX_Class * pObj = new XXX_Class(); 中的new我們一般稱其為C++關鍵字 (keyword), 就以這個語句為例做了兩個操作:

調用了operator new從堆上申請所需的空間

調用XXX_Class的構造函數

那么當你調用delete pObj;的時候,道理同new,剛好相反:

調用了XXX_Class的析構函數

通過operator delete 釋放了內存

一切似乎都沒有什么問題,然后又一個坑來了。但如果申請的是一個數組呢,入下述例子:

class MemoryLeakClass

{public:

MemoryLeakClass()

{

m_pStr = new char[100];

}

void DoSomething()

{

strcpy_s(m_pStr, 100, “Hello Memory Leak!”);

std::cout 《《 m_pStr 《《 std::endl;

}

~MemoryLeakClass()

{

delete m_pStr;

}

private:

char *m_pStr;

};

void MemoryLeakFunction()

{

const int iSize = 5;

MemoryLeakClass* pArrayObjs = new MemoryLeakClass [iSize];

for (int i = 0; i 《 iSize; i++)

{

(pArrayObjs+i)-》DoSomething();

}

delete pArrayObjs;

}

上述例子通過MemoryLeakClass* pArrayObjs = new MemoryLeakClass [iSize];申請了一個MemoryLeakClass數組,那么調用不匹配的delete pArrayObjs;, 會產生內存泄露。先看看下圖, 然后結合剛講的delete的行為:

那么其實調用delete pArrayObjs;的時候,釋放了整個pArrayObjs的內存,但是只調用了pArrayObjs[0]析構函數并釋放中的m_pStr指向的內存。pArrayObjs 1~4并沒有調用析構函數,從而導致其中的m_pStr指向的內存沒有釋放。所以我們要注意new和delete要匹配使用,當使用的new []申請的內存最好要用delete[]。那么留一個問題給讀者, 上面代碼delete m_pStr;會導致同樣的問題嗎?如果總是要讓我們自己去保證,new和delete的配對,顯然還是難以避免錯誤的發生的。這個時候也可以使用unique_ptr, 修改如下:

void MemoryLeakFunction()

{

const int iSize = 5;

std::unique_ptr《MemoryLeakClass[]》 pArrayObjs = std::make_unique《MemoryLeakClass[]》(iSize);

for (int i = 0; i 《 iSize; i++)

{

(pArrayObjs.get()+i)-》DoSomething();

}

}

3. delete (void*)

如果上一個章節已經有理解,那么對于這個例子,就很容易明白了。正因為C++的靈活性,有時候會將一個對象指針轉換為void *,隱藏其類型。這種情況SDK比較常用,實際上返回的并不是SDK用的實際類型,而是一個沒有類型的地址,當然有時候我們會為其親切的取一個名字,比如叫做XXX_HANDLE。那么繼續用上述為例MemoryLeakClass, SDK假設提供了下面三個接口

InitObj創建一個對象,并且返回一個PROGRAMER_HANDLE(即void *),對應用程序屏蔽其實際類型

DoSomething 提供了一個功能去做一些事情,輸入的參數,即為通過InitObj申請的對象

應用程序使用完畢后,一般需要釋放SDK申請的對象,提供了FreeObj

typedef void * PROGRAMER_HANDLE;

PROGRAMER_HANDLE InitObj()

{

MemoryLeakClass* pObj = new MemoryLeakClass();

return (PROGRAMER_HANDLE)pObj;

}

void DoSomething(PROGRAMER_HANDLE pHandle)

{

((MemoryLeakClass*)pHandle)-》DoSomething();

}

void FreeObj(void *pObj)

{

delete pObj;

}

看到這里,也許有讀者已經發現問題所在了。上述代碼在調用FreeObj的時候,delete看到的是一個void *, 只會釋放對象所占用的內存,但是并不會調用對象的析構函數,那么對象內部的m_pStr所指向的內存并沒有被釋放,從而會導致內存泄露。修改也是自然比較簡單的:

void FreeObj(void *pObj)

{

delete ((MemoryLeakClass*)pObj);

}

那么一般來說,最好由相對資深的程序員去進行SDK的開發,無論從設計和實現上面,都盡量避免了各種讓人淚流滿滿的坑。

4. Virtual destructor

現在大家來看看這個很容易犯錯的場景, 一個很常用的多態場景。那么在調用delete pObj;會出現內存泄露嗎?

class Father

{public:

virtual void DoSomething()

{

std::cout 《《 “Father DoSomething()” 《《 std::endl;

}

};

class Child : public Father

{

public:

Child()

{

std::cout 《《 “Child()” 《《 std::endl;

m_pStr = new char[100];

}

~Child()

{

std::cout 《《 “~Child()” 《《 std::endl;

delete[] m_pStr;

}

void DoSomething()

{

std::cout 《《 “Child DoSomething()” 《《 std::endl;

}

protected:

char* m_pStr;

};

void MemoryLeakVirualDestructor()

{

Father * pObj = new Child;

pObj-》DoSomething();

delete pObj;

}

會的,因為Father沒有設置Virtual 析構函數,那么在調用delete pObj;的時候會直接調用Father的析構函數,而不會調用Child的析構函數,這就導致了Child中的m_pStr所指向的內存,并沒有被釋放,從而導致了內存泄露。并不是絕對,當有這種使用場景的時候,最好是設置基類的析構函數為虛析構函數。修改如下:

class Father

{public:

virtual void DoSomething()

{

std::cout 《《 “Father DoSomething()” 《《 std::endl;

}

virtual ~Father() { ; }

};

class Child : public Father

{

public:

Child()

{

std::cout 《《 “Child()” 《《 std::endl;

m_pStr = new char[100];

}

virtual ~Child()

{

std::cout 《《 “~Child()” 《《 std::endl;

delete[] m_pStr;

}

void DoSomething()

{

std::cout 《《 “Child DoSomething()” 《《 std::endl;

}

protected:

char* m_pStr;

};

5. 對象循環引用

看下面例子,既然為了防止內存泄露,于是使用了智能指針shared_ptr;并且這個例子就是創建了一個雙向鏈表,為了簡單演示,只有兩個節點作為演示,創建了鏈表后,對鏈表進行遍歷。

那么這個例子會導致內存泄露嗎?

struct Node

{

Node(int iVal)

{

m_iVal = iVal;

}

~Node()

{

std::cout 《《 “~Node(): ” 《《 “Node Value: ” 《《 m_iVal 《《 std::endl;

}

void PrintNode()

{

std::cout 《《 “Node Value: ” 《《 m_iVal 《《 std::endl;

}

std::shared_ptr《Node》 m_pPreNode;

std::shared_ptr《Node》 m_pNextNode;

int m_iVal;

};

void MemoryLeakLoopReference()

{

std::shared_ptr《Node》 pFirstNode = std::make_shared《Node》(100);

std::shared_ptr《Node》 pSecondNode = std::make_shared《Node》(200);

pFirstNode-》m_pNextNode = pSecondNode;

pSecondNode-》m_pPreNode = pFirstNode;

//Iterate nodes

auto pNode = pFirstNode;

while (pNode)

{

pNode-》PrintNode();

pNode = pNode-》m_pNextNode;

}

}

先來看看下圖,是鏈表創建完成后的示意圖。有點暈乎了,怎么一個雙向鏈表畫的這么復雜,黃色背景的均為智能指針或者智能指針的組成部分。其實根據雙向鏈表的簡單性和下圖的復雜性,可以想到,智能指針的引入雖然提高了安全性,但是損失的是性能。所以往往安全性和性能是需要互相權衡的。 我們繼續往下看,哪里內存泄露了呢?

如果函數退出,那么m_pFirstNode和m_pNextNode作為棧上局部變量,智能指針本身調用自己的析構函數,給引用的對象引用計數減去1(shared_ptr本質采用引用計數,當引用計數為0的時候,才會刪除對象)。此時如下圖所示,可以看到智能指針的引用計數仍然為1, 這也就導致了這兩個節點的實際內存,并沒有被釋放掉, 從而導致內存泄露。

你可以在函數返回前手動調用pFirstNode-》m_pNextNode.reset();強制讓引用計數減去1, 打破這個循環引用。

還是之前那句話,如果通過手動去控制難免會出現遺漏的情況, C++提供了weak_ptr。

struct Node

{

Node(int iVal)

{

m_iVal = iVal;

}

~Node()

{

std::cout 《《 “~Node(): ” 《《 “Node Value: ” 《《 m_iVal 《《 std::endl;

}

void PrintNode()

{

std::cout 《《 “Node Value: ” 《《 m_iVal 《《 std::endl;

}

std::shared_ptr《Node》 m_pPreNode;

std::weak_ptr《Node》 m_pNextNode;

int m_iVal;

};

void MemoryLeakLoopRefference()

{

std::shared_ptr《Node》 pFirstNode = std::make_shared《Node》(100);

std::shared_ptr《Node》 pSecondNode = std::make_shared《Node》(200);

pFirstNode-》m_pNextNode = pSecondNode;

pSecondNode-》m_pPreNode = pFirstNode;

//Iterate nodes

auto pNode = pFirstNode;

while (pNode)

{

pNode-》PrintNode();

pNode = pNode-》m_pNextNode.lock();

}

}

看看使用了weak_ptr之后的鏈表結構如下圖所示,weak_ptr只是對管理的對象做了一個弱引用,其并不會實際支配對象的釋放與否,對象在引用計數為0的時候就進行了釋放,而無需關心weak_ptr的weak計數。注意shared_ptr本身也會對weak計數加1.

那么在函數退出后,當pSecondNode調用析構函數的時候,對象的引用計數減一,引用計數為0,釋放第二個Node,在釋放第二個Node的過程中又調用了m_pPreNode的析構函數,第一個Node對象的引用計數減1,再加上pFirstNode析構函數對第一個Node對象的引用計數也減去1,那么第一個Node對象的引用計數也為0,第一個Node對象也進行了釋放。

如果將上述代碼改為雙向循環鏈表,去除那個循環遍歷Node的代碼,那么最后Node的內存會被釋放嗎?這個問題留給讀者。

6. 資源泄露

如果說些作文的話,這一章節,可能有點偏題了。本章要講的是廣義上的資源泄露,比如句柄或者fd泄露。這些也算是內存泄露的一點點擴展,寫作文的一點點延伸吧。

看看下述例子, 其在操作完文件后,忘記調用CloseHandle(hFile);了,從而導致內存泄露。

void MemroyLeakFileHandle()

{

HANDLE hFile = CreateFile(LR“(C: estdoc.txt)”,

GENERIC_READ

FILE_SHARE_READ,

NULL,

OPEN_EXISTING,

FILE_ATTRIBUTE_NORMAL,

NULL);

if (INVALID_HANDLE_VALUE == hFile)

{

std::cerr 《《 “Open File error!” 《《 std::endl;

return;

}

const int BUFFER_SIZE = 100;

char pDataBuffer[BUFFER_SIZE];

DWORD dwBufferSize;

if (ReadFile(hFile,

pDataBuffer,

BUFFER_SIZE,

&dwBufferSize,

NULL))

{

std::cout 《《 dwBufferSize 《《 std::endl;

}

}

上述你可以用RAII機制去封裝hFile從而讓其在函數退出后,直接調用CloseHandle(hFile);。C++智能指針提供了自定義deleter的功能,這就可以讓我們使用這個deleter的功能,改寫代碼如下。不過本人更傾向于使用類似于golang defer的實現方式,讀者可以參閱本文相關閱讀部分。

void MemroyLeakFileHandle()

{

HANDLE hFile = CreateFile(LR“(C: estdoc.txt)”,

GENERIC_READ,

FILE_SHARE_READ,

NULL,

OPEN_EXISTING,

FILE_ATTRIBUTE_NORMAL,

NULL);

std::unique_ptr《 HANDLE, std::function《void(HANDLE*)》》 phFile(

&hFile,

[](HANDLE* pHandle) {

if (nullptr != pHandle)

{

std::cout 《《 “Close Handle” 《《 std::endl;

CloseHandle(*pHandle);

}

});

if (INVALID_HANDLE_VALUE == *phFile)

{

std::cerr 《《 “Open File error!” 《《 std::endl;

return;

}

const int BUFFER_SIZE = 100;

char pDataBuffer[BUFFER_SIZE];

DWORD dwBufferSize;

if (ReadFile(*phFile,

pDataBuffer,

BUFFER_SIZE,

&dwBufferSize,

NULL))

{

std::cout 《《 dwBufferSize 《《 std::endl;

}

}

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 內存
    +關注

    關注

    8

    文章

    3055

    瀏覽量

    74338
  • 函數
    +關注

    關注

    3

    文章

    4346

    瀏覽量

    62979
  • C++
    C++
    +關注

    關注

    22

    文章

    2114

    瀏覽量

    73860

原文標題:6. 資源泄露

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Spire.XLS for C++組件說明

    Spire.XLS for C++ 是一款專業的 C++ Excel 組件,可以用在各種 C++ 框架和應用程序。Spire.XLS for C+
    的頭像 發表于 01-14 09:40 ?150次閱讀
    Spire.XLS for <b class='flag-5'>C++</b>組件說明

    EE-112:模擬C++的類實現

    電子發燒友網站提供《EE-112:模擬C++的類實現.pdf》資料免費下載
    發表于 01-03 15:15 ?0次下載
    EE-112:模擬<b class='flag-5'>C++</b><b class='flag-5'>中</b>的類實現

    C7000 C/C++優化指南用戶手冊

    電子發燒友網站提供《C7000 C/C++優化指南用戶手冊.pdf》資料免費下載
    發表于 11-09 15:00 ?0次下載
    <b class='flag-5'>C</b>7000 <b class='flag-5'>C</b>/<b class='flag-5'>C++</b>優化指南用戶手冊

    TMS320C6000優化C/C++編譯器v8.3.x

    電子發燒友網站提供《TMS320C6000優化C/C++編譯器v8.3.x.pdf》資料免費下載
    發表于 11-01 09:35 ?0次下載
    TMS320<b class='flag-5'>C</b>6000優化<b class='flag-5'>C</b>/<b class='flag-5'>C++</b>編譯器v8.3.x

    C語言和C++結構體的區別

    同樣是結構體,看看在C語言和C++中有什么區別?
    的頭像 發表于 10-30 15:11 ?357次閱讀

    C7000優化C/C++編譯器

    電子發燒友網站提供《C7000優化C/C++編譯器.pdf》資料免費下載
    發表于 10-30 09:45 ?0次下載
    <b class='flag-5'>C</b>7000優化<b class='flag-5'>C</b>/<b class='flag-5'>C++</b>編譯器

    使用OpenVINO GenAI API在C++構建AI應用程序

    許多桌面應用程序是使用 C++ 開發的,而將生成式AI(GenAI)功能集成到這些應用程序可能會很具有挑戰性,尤其是因為使用像 Hugging Face 這樣的 Python 庫的復雜性。C++
    的頭像 發表于 10-12 09:36 ?471次閱讀
    使用OpenVINO GenAI API在<b class='flag-5'>C++</b><b class='flag-5'>中</b>構建AI應用程序

    ostream在c++的用法

    ostream 是 C++ 標準庫中一個非常重要的類,它位于 頭文件(實際上,更常見的是通過包含 頭文件來間接包含 ,因為 包含了 和 )。 ostream 類及其派生類(如 std::cout
    的頭像 發表于 09-20 15:11 ?952次閱讀

    OpenVINO2024 C++推理使用技巧

    很多人都使用OpenVINO新版的C++ 或者Python的SDK,都覺得非常好用,OpenVINO2022之后的版本C++ SDK做了大量的優化與整理,已經是非常貼近開發的使用習慣與推理方式。與OpenCV的Mat對象對接方式更是幾乎無縫對接,非常的方便好用。
    的頭像 發表于 07-26 09:20 ?1057次閱讀

    ModusToolbox 3.2在c代碼包含c++代碼的正確步驟是什么?

    使用 ModusToolbox 3.2 我有一個用純 C 語言編寫的 XMC4700 項目。 我正在嘗試添加一些 C++ 函數,并將其合并到我的原始代碼。 我可以構建獨立的 .cpp/.hpp
    發表于 07-23 08:21

    C++語言基礎知識

    電子發燒友網站提供《C++語言基礎知識.pdf》資料免費下載
    發表于 07-19 10:58 ?7次下載

    C++實現類似instanceof的方法

    函數,可實際上C++沒有。但是別著急,其實C++中有兩種簡單的方法可以實現類似Java的instanceof的功能。 在 C++
    的頭像 發表于 07-18 10:16 ?673次閱讀
    <b class='flag-5'>C++</b><b class='flag-5'>中</b>實現類似instanceof的方法

    C/C++兩種宏實現方式

    #ifndef的方式受C/C++語言標準支持。它不僅可以保證同一個文件不會被包含多次,也能保證內容完全相同的兩個文件(或者代碼片段)不會被不小心同時包含。
    的頭像 發表于 04-19 11:50 ?702次閱讀

    鴻蒙OS開發實例:【Native C++

    使用DevEco Studio創建一個Native C++應用。應用采用Native C++模板,實現使用NAPI調用C標準庫的功能。使用C標準庫hypot接口計算兩個給定數平方和的平
    的頭像 發表于 04-14 11:43 ?2767次閱讀
    鴻蒙OS開發實例:【Native <b class='flag-5'>C++</b>】

    使用 MISRA C++:2023? 避免基于范圍的 for 循環中的錯誤

    在前兩篇博客,我們?向您介紹了新的 MISRA C++ 標準?和?C++ 的歷史?。在這篇博客,我們將仔細研究以 C++
    的頭像 發表于 03-28 13:53 ?861次閱讀
    使用 MISRA <b class='flag-5'>C++</b>:2023? 避免基于范圍的 for 循環中的錯誤
    主站蜘蛛池模板: 色综合天天综合给合国产 | 午夜影院普通 | 天天性综合 | 康熙古代高h细节肉爽文全文 | 天天曰 | 欧美人与动性视频在线观 | 亚洲天堂一区二区三区 | 快乐你懂的在线视频免费观看 | 午夜视频www | 久操精品在线观看 | 日本xxxxx黄区免费看动漫 | 美女鲜嫩bbbb | 欧美三级在线 | 四虎影在永久地址在线观看 | 色香视频一sxmv首页 | 伊人色强在线网 | 日本黄色免费大片 | 一级看片| 男女视频免费观看 | 2021国产精品久久 | 久久综合一 | 日本在线不卡免 | qvod高清在线成人观看 | 手机看片免费永久在线观看 | 国产成人a | 日本一区二区三区不卡在线视频 | 午夜寂寞视频在线观看 | 8050午夜一级二级全黄 | 色播五月激情 | 日本一区二区高清免费不卡 | 大香交伊人 | 狠狠色噜噜综合社区 | 国产片无遮挡在线看床戏 | 欧美精品二区 | www深夜视频在线观看高清 | 华人永久免费视频 | 天天躁夜夜躁狠狠躁2021西西 | 六月丁香婷婷天天在线 | 四虎成人在线视频 | 日本高清一区二区三区不卡免费 | 午夜日韩在线 |