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++語(yǔ)言里有哪些比較常見(jiàn)的坑?

Q4MP_gh_c472c21 ? 來(lái)源:程序喵大人 ? 作者:程序喵大人 ? 2021-05-20 11:38 ? 次閱讀

前段時(shí)間給部門(mén)做了個(gè)C++專(zhuān)題的分享,主要分享了C++語(yǔ)言里一些常見(jiàn)的坑,在這里也分享給大家。

首先說(shuō)下C++和C語(yǔ)言有什么區(qū)別?分享一個(gè)我在知乎上看見(jiàn)的回答:

C++ ≈ C with classes, C with STL

C:面向機(jī)器編程

C++:面向編譯器編程

C++有個(gè)很重要的特性叫RAII,個(gè)人認(rèn)為可以多多使用,相當(dāng)方便,關(guān)于RAII巧妙使用可以看我這兩篇文章《RAII妙用之ScopeExit》《RAII妙用之計(jì)算函數(shù)耗時(shí)》。

言歸正傳,下面我一個(gè)一個(gè)的列出來(lái)C++使用過(guò)程中常見(jiàn)的坑:

無(wú)符號(hào)整數(shù)的錯(cuò)誤使用

for (unsigned int i = 10; i 》= 0; --i) { 。.. }

上面這段代碼會(huì)發(fā)生什么? 會(huì)死循環(huán),這里要注意下無(wú)符號(hào)整數(shù)的使用。

容器的size()返回類(lèi)型是無(wú)符號(hào)整數(shù)

std::vector《int》 vec;vec.push_back(1);for (auto idx = vec.size(); idx 》= 0; idx--) { cout 《《 “=====

”;}

這段代碼依舊會(huì)出現(xiàn)死循環(huán),原因參考上一條。

memcpy、memset只適用于POD結(jié)構(gòu)

至于什么是POD類(lèi)型,其實(shí)解釋起來(lái)挺麻煩的,感興趣的可以直接看cppreference的https://en.cppreference.com/w/cpp/named_req/PODType

STL遍歷刪除時(shí)注意迭代器失效問(wèn)題

void erase(std::vector《int》 &vec, int a) { for (auto iter = vec.begin(); iter != vec.end();) { // 這個(gè)正確 if (*iter == a) { iter = vec.erase(iter); } else { ++iter; } }

for (auto iter = vec.begin(); iter != vec.end(); ++iter) { // error if (*iter == a) { vec.erase(iter); // error } }}

std::list排序使用自己的成員方法

一般的容器排序都使用std::sort(),但是list特殊。

int main() { std::list《int》 list{1, 2, 3, 2}; list.sort(); // std::sort(list.begin(), list.end()); for (auto i : list) { std::cout 《《 i 《《 “ ”; } std::cout 《《 “

”; return 0;}

new/delete、new[]/delete[]、malloc/free嚴(yán)格配對(duì)

這幾個(gè)一定要配對(duì)使用,原因的話可以看我之前的文章《new[]和delete[]為何要配對(duì)使用?》

基類(lèi)析構(gòu)函數(shù)要是虛函數(shù)

如果不是虛函數(shù)的話,可能會(huì)有內(nèi)存泄漏的問(wèn)題

注釋用/**/,而不是//

注釋用/**/,可能會(huì)出問(wèn)題。原因:utf-8和ANSC(GB2312)編碼混亂后,中文注釋就亂碼了,亂碼中藏著 */,匹配錯(cuò)了,導(dǎo)致IDE實(shí)際注釋的部分并非肉眼所見(jiàn),定位極其困難,常見(jiàn)于Windows中。

成員變量初始化

成員變量沒(méi)有默認(rèn)初始化行為,需要手動(dòng)初始化。

不要返回局部變量的指針或引用

char* func() { char a[3] = {‘a(chǎn)’, ‘b’, ‘c’}; return a;}

棧內(nèi)存容易被污染。

浮點(diǎn)數(shù)判斷是否相等問(wèn)題

float f;if (f == 0.2) {} // 錯(cuò)誤用法if (abs(f - 0.2) 《 0.00001) {} // 正確用法

vector clear和swap問(wèn)題

清空某個(gè)vector,可以使用swap而不是其clear方法,這樣可以更早的釋放vector內(nèi)部?jī)?nèi)存。

vector《int》 vec;vector《int》().swap(vec);vec.clear();

vector問(wèn)題

盡量不要在vector中存放bool類(lèi)型,vector為了做優(yōu)化,它的內(nèi)部存放的其實(shí)不是bool。

條件變量

條件變量的使用有兩大問(wèn)題:信號(hào)丟失和虛假喚醒,相當(dāng)重要,具體可以看我這篇文章《使用條件變量的坑你知道嗎》。

類(lèi)型轉(zhuǎn)換

在C++中盡量使用C++風(fēng)格的四種類(lèi)型轉(zhuǎn)換,而不要使用C語(yǔ)言風(fēng)格的強(qiáng)制類(lèi)型轉(zhuǎn)換。

異步操作中async的使用

std::async(std::async, []{ f(); }); // 臨時(shí)量的析構(gòu)函數(shù)等待 f()std::async(std::async, []{ g(); }); // f() 完成前不開(kāi)始

std::async 這貨返回的 future 和通過(guò) promise 獲取的 future 行為不同,async 返回的 future 對(duì)象在析構(gòu)時(shí)會(huì)阻塞等待 async 中的線程執(zhí)行完畢,這就導(dǎo)致在大部分場(chǎng)景中 async達(dá)不到你直覺(jué)的認(rèn)為它能達(dá)到的目的。

智能指針

一個(gè)裸指針不要使用多個(gè)智能指針包裹,盡可能使用make_unique,make_shared。

當(dāng)需要在類(lèi)得內(nèi)部接口中,需要將this作為智能指針使用,需要用該類(lèi)派生自enable_shared_from_this

棧內(nèi)存使用

合理使用棧內(nèi)存,特別是數(shù)組,數(shù)組越界問(wèn)題容易導(dǎo)致??臻g損壞,可以考慮使用std::array替代普通的數(shù)組。

std::thread的使用

一定要記得join或這detach,否則會(huì)crash。

void func() {}int main() { std::thread t(func); if (t.joinable()) { t.join(); // 或者t.detach(); } return 0;}

enum使用

盡量使用enum class替代enum,enum class 是帶有作用域的枚舉類(lèi)型。

空指針使用nullptr而不是NULL

至于為什么要這么使用,可以看我這篇文章《關(guān)于nullptr這篇文章你一定要看》

void func(char*) { cout 《《 “char*”;}void func(int) { cout 《《 “int”;}

int main() { func(NULL); // 編譯失敗 error: call of overloaded ‘func(NULL)’ is ambiguous func(nullptr); // char* return 0;}

std::remove的使用

這個(gè)remove其實(shí)并沒(méi)有真正的刪除元素,需要和erase配合使用,跑一下這段代碼就知道啦。

bool isOdd(int i) { return i & 1; }

void print(const std::vector《int》& vec) { for (const auto& i : vec) { std::cout 《《 i 《《 ‘ ’; } std::cout 《《 std::endl;}

int main() { std::vector《int》 v = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; print(v);

std::remove(v.begin(), v.end(), 5); // error print(v);

v.erase(std::remove(v.begin(), v.end(), 5), v.end()); print(v);

v.erase(std::remove_if(v.begin(), v.end(), isOdd), v.end()); print(v);}

全局變量初始化問(wèn)題

不同文件中的全局變量初始化順序不固定,全局變量盡量不要互相依賴(lài),否則由于初始化順序不固定的問(wèn)題,可能會(huì)導(dǎo)致bug產(chǎn)生。

原文標(biāo)題:C++的24個(gè)大坑,你能躲過(guò)幾個(gè)?

文章出處:【微信公眾號(hào):嵌入式ARM】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

責(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)投訴
  • 編程
    +關(guān)注

    關(guān)注

    88

    文章

    3544

    瀏覽量

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

    關(guān)注

    21

    文章

    2090

    瀏覽量

    73419

原文標(biāo)題:C++的24個(gè)大坑,你能躲過(guò)幾個(gè)?

文章出處:【微信號(hào):gh_c472c2199c88,微信公眾號(hào):嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    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的方法

    C++多態(tài)與繼承,但是很多人開(kāi)始學(xué)習(xí)C++,有時(shí)候會(huì)面臨一個(gè)常見(jiàn)問(wèn)題,就是如何向下轉(zhuǎn)型,特別是不知道具體類(lèi)型的時(shí)候,這個(gè)時(shí)候就希望C++
    的頭像 發(fā)表于 07-18 10:16 ?444次閱讀
    <b class='flag-5'>C++</b>中實(shí)現(xiàn)類(lèi)似instanceof的方法

    為什么很少用C++開(kāi)發(fā)單片機(jī)

    C語(yǔ)言是面向過(guò)程的語(yǔ)言,C++是面向?qū)ο蟮木幊?b class='flag-5'>語(yǔ)言。結(jié)合本文來(lái)說(shuō),面向過(guò)程相比面向?qū)ο蟮木幊?,生成代碼量(bin文件)更小,運(yùn)行效率更高。
    發(fā)表于 03-25 14:26 ?748次閱讀
    為什么很少用<b class='flag-5'>C++</b>開(kāi)發(fā)單片機(jī)

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

    C語(yǔ)言、C++、Java和Python是四種常見(jiàn)的編程語(yǔ)言,各有優(yōu)點(diǎn)和特點(diǎn)。 C
    的頭像 發(fā)表于 02-05 14:11 ?1684次閱讀

    vb語(yǔ)言c++語(yǔ)言的區(qū)別

    VB語(yǔ)言C++語(yǔ)言是兩種不同的編程語(yǔ)言,雖然它們都屬于高級(jí)編程語(yǔ)言,但在設(shè)計(jì)和用途上有很多區(qū)別。下面將詳細(xì)
    的頭像 發(fā)表于 02-01 10:20 ?1813次閱讀

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

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

    嵌入式工程師C語(yǔ)言小技巧

    軟件界看起來(lái)不怎么高大上,但是 MCU 的開(kāi)發(fā)目前 C 語(yǔ)言還是主流,為了更好的操控和調(diào)試我們的硬件,我們還是需要竭力的避免一些 C語(yǔ)言編程的陷阱,避免被一些高大上的變成
    的頭像 發(fā)表于 12-13 11:32 ?435次閱讀

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

    C語(yǔ)言雖說(shuō)經(jīng)常和C++在一起被大家提起,但可千萬(wàn)不要以為它們是一個(gè)東西。現(xiàn)在我們常用的C語(yǔ)言C
    的頭像 發(fā)表于 12-07 14:29 ?858次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>和<b class='flag-5'>C++</b>中那些不同的地方

    如何選擇創(chuàng)建c語(yǔ)言c++

    選擇創(chuàng)建 C 語(yǔ)言C++ 都需要綜合考慮多個(gè)因素。在決定使用哪種語(yǔ)言之前,我們需要對(duì)這兩種語(yǔ)言的特點(diǎn)、優(yōu)缺點(diǎn)、適用場(chǎng)景、學(xué)習(xí)成本等進(jìn)行全
    的頭像 發(fā)表于 11-27 15:58 ?533次閱讀

    visualc++怎么新建c語(yǔ)言文件

    Visual C++ 是一個(gè)集成開(kāi)發(fā)環(huán)境(IDE),用于開(kāi)發(fā) CC++ 程序。在 Visual C++ 中,新建一個(gè) C
    的頭像 發(fā)表于 11-27 15:57 ?3310次閱讀

    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ā)。
    的頭像 發(fā)表于 11-27 15:56 ?830次閱讀

    C/C++語(yǔ)言學(xué)習(xí)大全套

    電子發(fā)燒友網(wǎng)站提供《C/C++語(yǔ)言學(xué)習(xí)大全套.rar》資料免費(fèi)下載
    發(fā)表于 11-18 14:33 ?2次下載
    <b class='flag-5'>C</b>/<b class='flag-5'>C++</b><b class='flag-5'>語(yǔ)言</b>學(xué)習(xí)大全套

    C語(yǔ)言常見(jiàn)問(wèn)題集

    電子發(fā)燒友網(wǎng)站提供《C語(yǔ)言常見(jiàn)問(wèn)題集.pdf》資料免費(fèi)下載
    發(fā)表于 11-18 10:29 ?0次下載
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>的<b class='flag-5'>常見(jiàn)</b>問(wèn)題集

    單片機(jī)編程語(yǔ)言可以用c++嗎?

    單片機(jī)編程語(yǔ)言可以用c++嗎?
    發(fā)表于 11-09 06:40

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

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