成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

總結了24個C++的大坑,看你能躲過幾個?

網絡 通信技術
std::async 這貨返回的 future 和通過 promise 獲取的 future 行為不同,async 返回的 future 對象在析構時會阻塞等待 async 中的線程執行完畢,這就導致在大部分場景中 async達不到你直覺的認為它能達到的目的。

[[396092]]

本文轉載自微信公眾號「程序喵大人」,作者程序喵大人。轉載本文請聯系程序喵大人公眾號。

以下是本文目錄:

 

首先說下C++和C語言有什么區別?分享一個我在知乎上看見的回答:

  • C++ ≈ C with classes, C with STL
  • C:面向機器編程
  • C++:面向編譯器編程

C++有個很重要的特性叫RAII,個人認為可以多多使用,相當方便,關于RAII巧妙使用可以看我這兩篇文章《RAII妙用之ScopeExit》《RAII妙用之計算函數耗時》。

言歸正傳,下面我一個一個的列出來C++使用過程中常見的坑:

無符號整數的錯誤使用

  1. for (unsigned int i = 10; i >= 0; --i) { ... } 

上面這段代碼會發生什么? 會死循環,這里要注意下無符號整數的使用。

容器的size()返回類型是無符號整數

  1. std::vector<int> vec; 
  2. vec.push_back(1); 
  3. for (auto idx = vec.size(); idx >= 0; idx--) { 
  4.     cout << "===== \n"

這段代碼依舊會出現死循環,原因參考上一條。

memcpy、memset只適用于POD結構

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

STL遍歷刪除時注意迭代器失效問題

  1. void erase(std::vector<int> &vec, int a) { 
  2.     for (auto iter = vec.begin(); iter != vec.end();) { // 這個正確 
  3.         if (*iter == a) { 
  4.             iter = vec.erase(iter); 
  5.         } else { 
  6.             ++iter; 
  7.         } 
  8.     } 
  9.  
  10.     for (auto iter = vec.begin(); iter != vec.end(); ++iter) {  // error 
  11.         if (*iter == a) { 
  12.             vec.erase(iter); // error 
  13.         } 
  14.     } 

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

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

  1. int main() { 
  2.     std::list<int> list{1, 2, 3, 2}; 
  3.     list.sort(); 
  4.     // std::sort(list.begin(), list.end()); 
  5.     for (auto i : list) { 
  6.         std::cout << i << " "
  7.     } 
  8.     std::cout << "\n"
  9.     return 0; 

new/delete、new[]/delete[]、malloc/free嚴格配對

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

基類析構函數要是虛函數

如果不是虛函數的話,可能會有內存泄漏的問題

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

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

成員變量初始化

成員變量沒有默認初始化行為,需要手動初始化。

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

  1. char* func() { 
  2.     char a[3] = {'a''b''c'}; 
  3.     return a; 

棧內存容易被污染。

浮點數判斷是否相等問題

  1. float f; 
  2. if (f == 0.2) {} // 錯誤用法 
  3. if (abs(f - 0.2) < 0.00001) {} // 正確用法 

vector clear和swap問題

清空某個vector,可以使用swap而不是其clear方法,這樣可以更早的釋放vector內部內存。

  1. vector<int> vec; 
  2. vector<int>().swap(vec); 
  3. vec.clear(); 

vector問題

盡量不要在vector中存放bool類型,vector為了做優化,它的內部存放的其實不是bool。

條件變量

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

類型轉換

在C++中盡量使用C++風格的四種類型轉換,而不要使用C語言風格的強制類型轉換。

異步操作中async的使用

  1. std::async(std::launch::async, []{ f(); }); // 臨時量的析構函數等待 f() 
  2. std::async(std::launch::async, []{ g(); }); // f() 完成前不開始 

std::async 這貨返回的 future 和通過 promise 獲取的 future 行為不同,async 返回的 future 對象在析構時會阻塞等待 async 中的線程執行完畢,這就導致在大部分場景中 async達不到你直覺的認為它能達到的目的。

智能指針

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

當需要在類得內部接口中,需要將this作為智能指針使用,需要用該類派生自enable_shared_from_this

棧內存使用

合理使用棧內存,特別是數組,數組越界問題容易導致棧空間損壞,可以考慮使用std::array替代普通的數組。

std::thread的使用

一定要記得join或這detach,否則會crash。

  1. void func() {} 
  2. int main() { 
  3.     std::thread t(func); 
  4.     if (t.joinable()) { 
  5.         t.join(); // 或者t.detach();  
  6.     } 
  7.     return 0; 

enum使用

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

空指針使用nullptr而不是NULL

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

  1. void func(char*) { 
  2.     cout << "char*"
  3. void func(int) { 
  4.     cout << "int"
  5.  
  6. int main() { 
  7.      func(NULL); // 編譯失敗 error: call of overloaded ‘func(NULL)’ is ambiguous 
  8.     func(nullptr); // char
  9.     return 0; 

std::remove的使用

這個remove其實并沒有真正的刪除元素,需要和erase配合使用,跑一下這段代碼就知道啦。

  1. bool isOdd(int i) { return i & 1; } 
  2.  
  3. void print(const std::vector<int>& vec) { 
  4.     for (const auto& i : vec) { 
  5.         std::cout << i << ' '
  6.     } 
  7.     std::cout << std::endl; 
  8.  
  9. int main() { 
  10.     std::vector<int> v = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; 
  11.     print(v); 
  12.  
  13.     std::remove(v.begin(), v.end(), 5);  // error 
  14.     print(v); 
  15.  
  16.     v.erase(std::remove(v.begin(), v.end(), 5), v.end()); 
  17.     print(v); 
  18.  
  19.     v.erase(std::remove_if(v.begin(), v.end(), isOdd), v.end()); 
  20.     print(v); 

全局變量初始化問題

不同文件中的全局變量初始化順序不固定,全局變量盡量不要互相依賴,否則由于初始化順序不固定的問題,可能會導致bug產生。

 

責任編輯:武曉燕 來源: 程序喵大人
相關推薦

2021-02-22 07:58:47

內存程序變量

2023-08-28 12:09:53

2020-03-24 09:34:00

移動端H5軟鍵盤

2023-07-17 11:43:07

2021-11-19 16:54:11

Python代碼開發

2025-03-11 06:28:21

2024-02-21 14:55:19

C++語言編程

2011-05-07 14:39:00

投影

2019-07-16 13:59:43

數據庫MySQL軟件

2010-01-22 16:35:41

C++開發

2010-01-25 11:21:01

C++語法

2010-01-26 17:11:13

C++編程

2011-04-07 16:34:05

staticC++

2011-04-19 16:38:00

對象指針指針C++

2020-06-07 16:16:01

Python開發工具

2022-09-04 19:30:13

云原生系統

2020-06-10 10:30:48

Python 開發編程語言

2024-06-13 11:54:03

2024-04-03 12:30:00

C++開發

2022-03-21 14:09:19

面試C語言代碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美日韩视频在线第一区 | 伊人看片 | 国产午夜精品久久久 | 欧美亚洲另类丝袜综合网动图 | 午夜视频导航 | 久久com| 国产成人精品一区二区三区 | 夜夜爽99久久国产综合精品女不卡 | 一区二区三区视频在线观看 | 91精品欧美久久久久久久 | 成人免费av | 久久99精品久久久久婷婷 | 久久精品高清视频 | 做a视频在线观看 | 国产成人99久久亚洲综合精品 | 国产亚洲www | 日本欧美国产在线 | 在线观看视频91 | 国精产品一品二品国精在线观看 | 精品久久一区 | 粉嫩一区二区三区四区公司1 | 亚洲视频免费观看 | 天天躁日日躁狠狠躁2018小说 | 亚洲精品一区二区另类图片 | 欧美一区永久视频免费观看 | 国产欧美日韩一区二区三区在线 | 日韩欧美在线观看 | 日本在线一二 | www.日本国产 | 欧美日韩在线一区 | 免费一区二区三区 | 久久婷婷av | 亚洲区一区二区 | 综合天天久久 | 国产一区二区三区色淫影院 | 国产精品福利在线 | 看片91 | 欧美福利在线 | 美女久久 | 成人av播放 | 欧美一级在线观看 |