0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C++中的資源泄露問(wèn)題

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

在Modern C++之前,C++無(wú)疑是個(gè)更容易寫(xiě)出坑的語(yǔ)言,無(wú)論從開(kāi)發(fā)效率,和易坑性,讓很多新手望而卻步。比如內(nèi)存泄露問(wèn)題,就是經(jīng)常會(huì)被寫(xiě)出來(lái)的坑,本文就讓我們一起來(lái)看看,這些讓現(xiàn)在或者曾經(jīng)的C++程序員淚流滿面的內(nèi)存泄露場(chǎng)景吧。你是否有踩過(guò)?

1. 函數(shù)內(nèi)或者類(lèi)成員內(nèi)存未釋放

這類(lèi)問(wèn)題可以稱(chēng)之為out of scope的時(shí)候,并沒(méi)有釋放相應(yīng)對(duì)象的堆上內(nèi)存。有時(shí)候最簡(jiǎn)單的場(chǎng)景,反而是最容易犯錯(cuò)的。這個(gè)我想主要是因?yàn)榻?jīng)常寫(xiě),哪有不出錯(cuò)。下面場(chǎng)景一看就知道了,當(dāng)你在寫(xiě)XXX_Class * pObj = new XXX_Class();這一行的時(shí)候,腦子里面還在默念記得要釋放pObj ,記得要釋放pObj, 可能因?yàn)橹匾氖虑橐f(shuō)三遍,而你只喊了兩遍,最終還是忘記了寫(xiě)delete pObj; 這樣去釋放對(duì)象。

void MemoryLeakFunction()

{

XXX_Class * pObj = new XXX_Class();

pObj-》DoSomething();

return;

}

下面這個(gè)場(chǎng)景,就是析構(gòu)函數(shù)中并沒(méi)有釋放成員所指向的內(nèi)存。這個(gè)我們就要注意了,一般當(dāng)你構(gòu)建一個(gè)類(lèi)的時(shí)候,寫(xiě)析構(gòu)函數(shù)一定要切記釋放類(lèi)成員關(guān)聯(lián)的資源。

class MemoryLeakClass

{

public

MemoryLeakClass()

{

m_pObj = new XXX_ResourceClass;

}

void DoSomething()

{

m_pObj-》DoSomething();

}

~MemoryLeakClass()

{

;

}

private:

XXX_ResourceClass* m_pObj;

};

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

void MemoryLeakFunction()

{

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

pObj-》DoSomething();

return;

}

2. delete []

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

調(diào)用了operator new從堆上申請(qǐng)所需的空間

調(diào)用XXX_Class的構(gòu)造函數(shù)

那么當(dāng)你調(diào)用delete pObj;的時(shí)候,道理同new,剛好相反:

調(diào)用了XXX_Class的析構(gòu)函數(shù)

通過(guò)operator delete 釋放了內(nèi)存

一切似乎都沒(méi)有什么問(wèn)題,然后又一個(gè)坑來(lái)了。但如果申請(qǐng)的是一個(gè)數(shù)組呢,入下述例子:

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;

}

上述例子通過(guò)MemoryLeakClass* pArrayObjs = new MemoryLeakClass [iSize];申請(qǐng)了一個(gè)MemoryLeakClass數(shù)組,那么調(diào)用不匹配的delete pArrayObjs;, 會(huì)產(chǎn)生內(nèi)存泄露。先看看下圖, 然后結(jié)合剛講的delete的行為:

那么其實(shí)調(diào)用delete pArrayObjs;的時(shí)候,釋放了整個(gè)pArrayObjs的內(nèi)存,但是只調(diào)用了pArrayObjs[0]析構(gòu)函數(shù)并釋放中的m_pStr指向的內(nèi)存。pArrayObjs 1~4并沒(méi)有調(diào)用析構(gòu)函數(shù),從而導(dǎo)致其中的m_pStr指向的內(nèi)存沒(méi)有釋放。所以我們要注意new和delete要匹配使用,當(dāng)使用的new []申請(qǐng)的內(nèi)存最好要用delete[]。那么留一個(gè)問(wèn)題給讀者, 上面代碼delete m_pStr;會(huì)導(dǎo)致同樣的問(wèn)題嗎?如果總是要讓我們自己去保證,new和delete的配對(duì),顯然還是難以避免錯(cuò)誤的發(fā)生的。這個(gè)時(shí)候也可以使用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*)

如果上一個(gè)章節(jié)已經(jīng)有理解,那么對(duì)于這個(gè)例子,就很容易明白了。正因?yàn)镃++的靈活性,有時(shí)候會(huì)將一個(gè)對(duì)象指針轉(zhuǎn)換為void *,隱藏其類(lèi)型。這種情況SDK比較常用,實(shí)際上返回的并不是SDK用的實(shí)際類(lèi)型,而是一個(gè)沒(méi)有類(lèi)型的地址,當(dāng)然有時(shí)候我們會(huì)為其親切的取一個(gè)名字,比如叫做XXX_HANDLE。那么繼續(xù)用上述為例MemoryLeakClass, SDK假設(shè)提供了下面三個(gè)接口

InitObj創(chuàng)建一個(gè)對(duì)象,并且返回一個(gè)PROGRAMER_HANDLE(即void *),對(duì)應(yīng)用程序屏蔽其實(shí)際類(lèi)型

DoSomething 提供了一個(gè)功能去做一些事情,輸入的參數(shù),即為通過(guò)InitObj申請(qǐng)的對(duì)象

應(yīng)用程序使用完畢后,一般需要釋放SDK申請(qǐng)的對(duì)象,提供了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;

}

看到這里,也許有讀者已經(jīng)發(fā)現(xiàn)問(wèn)題所在了。上述代碼在調(diào)用FreeObj的時(shí)候,delete看到的是一個(gè)void *, 只會(huì)釋放對(duì)象所占用的內(nèi)存,但是并不會(huì)調(diào)用對(duì)象的析構(gòu)函數(shù),那么對(duì)象內(nèi)部的m_pStr所指向的內(nèi)存并沒(méi)有被釋放,從而會(huì)導(dǎo)致內(nèi)存泄露。修改也是自然比較簡(jiǎn)單的:

void FreeObj(void *pObj)

{

delete ((MemoryLeakClass*)pObj);

}

那么一般來(lái)說(shuō),最好由相對(duì)資深的程序員去進(jìn)行SDK的開(kāi)發(fā),無(wú)論從設(shè)計(jì)和實(shí)現(xiàn)上面,都盡量避免了各種讓人淚流滿滿的坑。

4. Virtual destructor

現(xiàn)在大家來(lái)看看這個(gè)很容易犯錯(cuò)的場(chǎng)景, 一個(gè)很常用的多態(tài)場(chǎng)景。那么在調(diào)用delete pObj;會(huì)出現(xiàn)內(nèi)存泄露嗎?

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;

}

會(huì)的,因?yàn)镕ather沒(méi)有設(shè)置Virtual 析構(gòu)函數(shù),那么在調(diào)用delete pObj;的時(shí)候會(huì)直接調(diào)用Father的析構(gòu)函數(shù),而不會(huì)調(diào)用Child的析構(gòu)函數(shù),這就導(dǎo)致了Child中的m_pStr所指向的內(nèi)存,并沒(méi)有被釋放,從而導(dǎo)致了內(nèi)存泄露。并不是絕對(duì),當(dāng)有這種使用場(chǎng)景的時(shí)候,最好是設(shè)置基類(lèi)的析構(gòu)函數(shù)為虛析構(gòu)函數(shù)。修改如下:

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. 對(duì)象循環(huán)引用

看下面例子,既然為了防止內(nèi)存泄露,于是使用了智能指針shared_ptr;并且這個(gè)例子就是創(chuàng)建了一個(gè)雙向鏈表,為了簡(jiǎn)單演示,只有兩個(gè)節(jié)點(diǎn)作為演示,創(chuàng)建了鏈表后,對(duì)鏈表進(jìn)行遍歷。

那么這個(gè)例子會(huì)導(dǎo)致內(nèi)存泄露嗎?

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;

}

}

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

如果函數(shù)退出,那么m_pFirstNode和m_pNextNode作為棧上局部變量,智能指針本身調(diào)用自己的析構(gòu)函數(shù),給引用的對(duì)象引用計(jì)數(shù)減去1(shared_ptr本質(zhì)采用引用計(jì)數(shù),當(dāng)引用計(jì)數(shù)為0的時(shí)候,才會(huì)刪除對(duì)象)。此時(shí)如下圖所示,可以看到智能指針的引用計(jì)數(shù)仍然為1, 這也就導(dǎo)致了這兩個(gè)節(jié)點(diǎn)的實(shí)際內(nèi)存,并沒(méi)有被釋放掉, 從而導(dǎo)致內(nèi)存泄露。

你可以在函數(shù)返回前手動(dòng)調(diào)用pFirstNode-》m_pNextNode.reset();強(qiáng)制讓引用計(jì)數(shù)減去1, 打破這個(gè)循環(huán)引用。

還是之前那句話,如果通過(guò)手動(dòng)去控制難免會(huì)出現(xiàn)遺漏的情況, 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之后的鏈表結(jié)構(gòu)如下圖所示,weak_ptr只是對(duì)管理的對(duì)象做了一個(gè)弱引用,其并不會(huì)實(shí)際支配對(duì)象的釋放與否,對(duì)象在引用計(jì)數(shù)為0的時(shí)候就進(jìn)行了釋放,而無(wú)需關(guān)心weak_ptr的weak計(jì)數(shù)。注意shared_ptr本身也會(huì)對(duì)weak計(jì)數(shù)加1.

那么在函數(shù)退出后,當(dāng)pSecondNode調(diào)用析構(gòu)函數(shù)的時(shí)候,對(duì)象的引用計(jì)數(shù)減一,引用計(jì)數(shù)為0,釋放第二個(gè)Node,在釋放第二個(gè)Node的過(guò)程中又調(diào)用了m_pPreNode的析構(gòu)函數(shù),第一個(gè)Node對(duì)象的引用計(jì)數(shù)減1,再加上pFirstNode析構(gòu)函數(shù)對(duì)第一個(gè)Node對(duì)象的引用計(jì)數(shù)也減去1,那么第一個(gè)Node對(duì)象的引用計(jì)數(shù)也為0,第一個(gè)Node對(duì)象也進(jìn)行了釋放。

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

6. 資源泄露

如果說(shuō)些作文的話,這一章節(jié),可能有點(diǎn)偏題了。本章要講的是廣義上的資源泄露,比如句柄或者fd泄露。這些也算是內(nèi)存泄露的一點(diǎn)點(diǎn)擴(kuò)展,寫(xiě)作文的一點(diǎn)點(diǎn)延伸吧。

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

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機(jī)制去封裝hFile從而讓其在函數(shù)退出后,直接調(diào)用CloseHandle(hFile);。C++智能指針提供了自定義deleter的功能,這就可以讓我們使用這個(gè)deleter的功能,改寫(xiě)代碼如下。不過(guò)本人更傾向于使用類(lèi)似于golang defer的實(shí)現(xiàn)方式,讀者可以參閱本文相關(guān)閱讀部分。

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;

}

}

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 內(nèi)存
    +關(guān)注

    關(guān)注

    8

    文章

    2903

    瀏覽量

    73536
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4237

    瀏覽量

    61967
  • C++
    C++
    +關(guān)注

    關(guān)注

    21

    文章

    2085

    瀏覽量

    73302

原文標(biāo)題:6. 資源泄露

文章出處:【微信號(hào):LinuxHub,微信公眾號(hào):Linux愛(ài)好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    ostream在c++的用法

    ostream 是 C++ 標(biāo)準(zhǔn)庫(kù)中一個(gè)非常重要的類(lèi),它位于 頭文件(實(shí)際上,更常見(jiàn)的是通過(guò)包含 頭文件來(lái)間接包含 ,因?yàn)?包含了 和 )。 ostream 類(lèi)及其派生類(lèi)(如 std::cout
    的頭像 發(fā)表于 09-20 15:11 ?95次閱讀

    C++語(yǔ)言基礎(chǔ)知識(shí)

    電子發(fā)燒友網(wǎng)站提供《C++語(yǔ)言基礎(chǔ)知識(shí).pdf》資料免費(fèi)下載
    發(fā)表于 07-19 10:58 ?6次下載

    C++實(shí)現(xiàn)類(lèi)似instanceof的方法

    函數(shù),可實(shí)際上C++沒(méi)有。但是別著急,其實(shí)C++中有兩種簡(jiǎn)單的方法可以實(shí)現(xiàn)類(lèi)似Java的instanceof的功能。 在 C++
    的頭像 發(fā)表于 07-18 10:16 ?355次閱讀
    <b class='flag-5'>C++</b><b class='flag-5'>中</b>實(shí)現(xiàn)類(lèi)似instanceof的方法

    C/C++兩種宏實(shí)現(xiàn)方式

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

    鴻蒙OS開(kāi)發(fā)實(shí)例:【Native C++

    使用DevEco Studio創(chuàng)建一個(gè)Native C++應(yīng)用。應(yīng)用采用Native C++模板,實(shí)現(xiàn)使用NAPI調(diào)用C標(biāo)準(zhǔn)庫(kù)的功能。使用C標(biāo)準(zhǔn)庫(kù)hypot接口計(jì)算兩個(gè)給定數(shù)平方和的平
    的頭像 發(fā)表于 04-14 11:43 ?2169次閱讀
    鴻蒙OS開(kāi)發(fā)實(shí)例:【Native <b class='flag-5'>C++</b>】

    使用 MISRA C++:2023? 避免基于范圍的 for 循環(huán)中的錯(cuò)誤

    在前兩篇博客,我們?向您介紹了新的 MISRA C++ 標(biāo)準(zhǔn)?和?C++ 的歷史?。在這篇博客,我們將仔細(xì)研究以 C++
    的頭像 發(fā)表于 03-28 13:53 ?517次閱讀
    使用 MISRA <b class='flag-5'>C++</b>:2023? 避免基于范圍的 for 循環(huán)中的錯(cuò)誤

    c語(yǔ)言,c++,java,python區(qū)別

    C語(yǔ)言、C++、Java和Python是四種常見(jiàn)的編程語(yǔ)言,各有優(yōu)點(diǎn)和特點(diǎn)。 C語(yǔ)言: C語(yǔ)言是一種面向過(guò)程的編程語(yǔ)言。它具有底層的特性,能夠?qū)τ?jì)算機(jī)硬件進(jìn)行直接操作。
    的頭像 發(fā)表于 02-05 14:11 ?1366次閱讀

    C++簡(jiǎn)史:C++是如何開(kāi)始的

    的 MISRA C++:2023 博客系列的第二部分。 在這篇博客,我們將深入探討 C++ 的歷史、編程語(yǔ)言多年來(lái)的發(fā)展歷程以及它的下一步發(fā)展方向。
    的頭像 發(fā)表于 01-11 09:00 ?427次閱讀
    <b class='flag-5'>C++</b>簡(jiǎn)史:<b class='flag-5'>C++</b>是如何開(kāi)始的

    C語(yǔ)言和C++那些不同的地方

    ++11標(biāo)準(zhǔn)。根據(jù)不同的標(biāo)準(zhǔn),它們的功能也會(huì)有所不同,但是越新的版本支持的編譯器越少,所以本文在討論的時(shí)候使用的C語(yǔ)言標(biāo)準(zhǔn)是C89,C++標(biāo)準(zhǔn)是C++99.我們來(lái)介紹
    的頭像 發(fā)表于 12-07 14:29 ?773次閱讀
    <b class='flag-5'>C</b>語(yǔ)言和<b class='flag-5'>C++</b><b class='flag-5'>中</b>那些不同的地方

    開(kāi)箱即用!教你如何正確使用華為云CodeArts IDE for C/C++

    C/C++編碼體驗(yàn)、方便的訪問(wèn)華為云資源、簡(jiǎn)單的引用華為云服務(wù)于一身,實(shí)現(xiàn)C/C++開(kāi)發(fā)者在個(gè)人研發(fā)作業(yè)體驗(yàn)和效率上的巨大提升。 為了幫助
    的頭像 發(fā)表于 11-29 17:40 ?611次閱讀
    開(kāi)箱即用!教你如何正確使用華為云CodeArts IDE for <b class='flag-5'>C</b>/<b class='flag-5'>C++</b>!

    c++怎么開(kāi)始編程

    C++是一種高級(jí)的、通用的編程語(yǔ)言,用于開(kāi)發(fā)各種類(lèi)型的應(yīng)用程序。它是從C語(yǔ)言演變而來(lái),也是一種靜態(tài)類(lèi)型語(yǔ)言,可以在不同的平臺(tái)上進(jìn)行開(kāi)發(fā)。C++具有高度的靈活性和性能,并且廣泛應(yīng)用于游戲開(kāi)發(fā)、桌面
    的頭像 發(fā)表于 11-27 15:56 ?736次閱讀

    c++多行注釋快捷鍵

    C++,多行注釋?zhuān)ㄒ卜Q(chēng)為塊注釋?zhuān)┦且环N用于注釋大段代碼或多個(gè)語(yǔ)句的方法。當(dāng)你希望暫時(shí)禁用一些代碼或者解釋特定部分代碼的作用時(shí),多行注釋是非常有用的。 在C++,多行注釋以 /*
    的頭像 發(fā)表于 11-22 10:24 ?6781次閱讀

    內(nèi)存是如何泄露

    作為 C++ 程序員,內(nèi)存泄露始終是懸在頭上的一顆炸彈。在過(guò)去幾年的 C++ 開(kāi)發(fā)過(guò)程,由于我們采用了一些技術(shù),我們的程序發(fā)生內(nèi)存泄露的情
    的頭像 發(fā)表于 11-13 14:13 ?348次閱讀
    內(nèi)存是如何<b class='flag-5'>泄露</b>的

    C++之父新作帶你勾勒現(xiàn)代C++地圖

    為了幫助大家解決這些痛點(diǎn)問(wèn)題,讓大家領(lǐng)略現(xiàn)代C++之美,掌握其中的精髓,更好地使用C++,C++之父Bjarne Stroustrup坐不住了,他親自操刀寫(xiě)就了這本《C++之旅》!
    的頭像 發(fā)表于 10-30 16:35 ?696次閱讀
    <b class='flag-5'>C++</b>之父新作帶你勾勒現(xiàn)代<b class='flag-5'>C++</b>地圖

    為什么會(huì)造成頻譜泄露?如何理解FFT的頻譜泄露效應(yīng)?

    為什么會(huì)造成頻譜泄露?如何理解FFT的頻譜泄露效應(yīng)?如何采用時(shí)間窗抑制頻譜泄露效應(yīng)? 頻譜泄露是指在進(jìn)行傅里葉變換時(shí),信號(hào)的譜被非目標(biāo)頻率
    的頭像 發(fā)表于 10-20 15:08 ?2534次閱讀