學過 C++ 的你,不得不知的這 10 條細節!

来源:https://www.cnblogs.com/xiaolincoding/archive/2020/02/12/12300745.html
-Advertisement-
Play Games

每日一句英語學習,每天進步一點點: “Action may not always bring happiness; but there is no happiness without action.” 「行動不見得一定帶來快樂,但沒有行動就沒有快樂。」 前言 我在閱讀 《Effective C++ ...



每日一句英語學習,每天進步一點點:
  • “Action may not always bring happiness; but there is no happiness without action.”
  • 「行動不見得一定帶來快樂,但沒有行動就沒有快樂。」

前言

我在閱讀 《Effective C++ (第三版本)》 書時做了不少筆記,從中收穫了非常多,也明白為什麼會書中前言的第一句話會說:

對於書中的「條款」這一詞,我更喜歡以「細節」替換,畢竟年輕的我們在打 LOL 或 王者的時,總會說註意細節!細節!細節~ —— 細節也算伴隨我們的青春的字眼

針對書中的前兩個章節,我篩選了 10 個 細節(條款)作為了本文的內容,這些細節也相對基礎且重要。

針對這 10 細節我都用較簡潔的例子來加以闡述,同時也把本文所提及細節中的「小結」總結繪畫成了一副思維導圖,便於大家的閱讀。

溫馨提示:本文較長(萬字),建議收藏閱讀。
後續有時間也會繼續分享後面章節的筆記,喜歡的小伙伴「點擊左上角」關註我~

正文

1 讓自己習慣C++

細節 01:儘量以const,enum,inline 替換 #define

#define 定義的常量有什麼不妥?

首先我們要清楚程式的編譯重要的三個階段:預處理階段,編譯階段和鏈接階段

#define 是不被視為語言的一部分,它在程式編譯階段中的預處理階段的作用,就是做簡單的替換。

如下麵的 PI 巨集定義,在程式編譯時,編譯器在預處理階段時,會先將源碼中所有 PI 巨集定義替換成 3.14

1#define PI 3.14

程式編譯在預處理階段後,才進行真正的編譯階段。在有的編譯器,運用了此 PI 常量,如果遇到了編譯錯誤,那麼這個錯誤信息也許會提到 3.14 而不是 PI,這就會讓人困惑哪裡來的3.14,特別是在項目大的情況下。

解決之道:以 const 定義一個常量替換上述的巨集(#define)

作為一個語言變數,下麵的 const 定義的常量 Pi 肯定會被編譯器看到,出錯的時候可以很清楚知道,是這個變數導致的問題:

1const doule Pi = 3.14;

如果是定義常量字元串,則必須要 const 兩次,目的是為了防止指針所指內容和指針自身不能被改變:

1const charconst myName = "小林coding";

如果是定義常量 string,則只需要在最前面加一次 const,形式如下:

1const std::string myName("小林coding");

#define 不重視作用域,所以對於 class 的專屬常量,應避免使用巨集定義。

還有另外一點巨集無法涉及的,就是我們無法利用 #define 創建一個 class 專屬常量,因為 #define 並不重視作用域。

對於類里要定義專屬常量時,我們依然使用 static + const,形式如下:

1class Student {
2private:
3    static const int num = 10;
4    int scores[num];
5};
6
7const int Student::num// static 成員變數,需要進行聲明

如果不想外部獲取到 class 專屬常量的記憶體地址,可以使用 enum 的方式定義常量

enum 會幫你約束這個條件,因為取一個 enum 的地址是不合法的,形式如下:

1class Student {
2private:
3    enum { num = 10 };
4    int scores[num];
5};

#define 實現的函數容易出錯,並且長相醜陋不易閱讀。

另外一個常見的 #define 誤用情況是以它實現巨集函數,它不會招致函數調用帶來的開銷,但是用 #define 編寫巨集函數容易出錯,如下用巨集定義寫的求最大值的函數:

1#define MAX(a, b) ( { (a) > (b) ? (a) : (b); } ) // 求最大值

這般長相的巨集有著太的缺點,比如在下麵調用例子:

1int a = 6, b = 5;
2int max = MAX(a++, b);
3
4std::cout << max << std::endl;
5std::cout << a << std::endl;

輸出結果(以下結果是錯誤的):

17 // 正確的答案是 max 輸出 6
28 // 正確的答案是  a  輸出 7

要解釋出錯的原因很簡單,我們把 MAX 巨集做簡單替換:

1int max = ( { (a++) > (b) ? (a++) : (b); } ); // a 被累加了2次!

在上述替換後,可以發現 a 被累加了 2 次。我們可以通過改進 MAX 巨集,來解決這個問題:

1#define MAX(a, b) ({ \
2    __typeof(a) __a = (a), __b = (b); \
3    __a > __b ? __a : __b; \
4})

簡單說明下,上述的 __typeof 可以根據變數的類型來定義一個相同類型的變數,如 a 變數是 int 類型,那麼 __a 變數的類型也是 int 類型。改進後的 MAX 巨集,輸出的是正確的結果,max 輸出 6,a 輸出 7。

雖然改進的後 MAX 巨集,解決了問題,但是這種巨集的長相就讓人困惑。

解決的方式:用 inline 替換 #define 定義的函數

inline 修飾的函數,也是可以解決函數調用的帶來的開銷,同時閱讀性較高,不會讓人困惑。

下麵用用 template inline 的方式,實現上述巨集定義的函數::

1template<typename T>
2inline T max(const T& a, const T& b)
3
{
4    return a > b? a : b;
5}

max 是一個真正的函數,它遵循作用域和訪問規則,所以不會出現變數被多次累加的現象。

模板的基礎知識記憶體,可移步到我的舊文進行學習 --> 泛型編程的第一步,掌握模板的特性!


細節 01 小結 - 請記住

  • 對於單純常量,最好以 const 對象或 enum 替換 #define;
  • 對於形式函數的巨集,最好改用 inline 函數替換 #define。

細節 02:儘可能使用 const

const 的一件奇妙的事情是:它允許你告訴編譯器和其他程式員某值應該保持不變

1. 面對指針,你可以指定指針自身、指針所指物,或兩者都(或都不)是 const:

1char myName[] = "小林coding";
2char *p = myName;             // non-const pointer, non-const data
3const char* p = myName;       // non-const pointer, const data
4charconst p = myName;       // const pointer, non-const data
5const charconst p = myName; // const pointer, const data
  • 如果關鍵詞const出現在星號(*邊,表示指針所指物是常量(不能改變 *p 的值);
  • 如果關鍵詞const出現在星號(*邊,表示指針自身是常量(不能改變 p 的值);
  • 如果關鍵詞const出現在星號(*邊,表示指針所指物和指針自身都是常量

2. 面對迭代器,你也指定迭代器自身或自迭代器所指物不可被改變:

1std::vector<int> vec;
2
3const std::vector<int>::iterator iter = vec.begin(); // iter 的作用像 T* const
4*iter = 10// 沒問題,可以改變 iter 所指物   
5++iter;     // 錯誤! 因為 iter 是 const     
6
7std::vector<int>::const_iterator cIter = vec.begin(); // cIter 的作用像 const T*
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 開發環境: Windows操作系統開發工具:Eclipse+Jdk+Tomcat+MySQL資料庫運行效果圖: 源碼及原文鏈接:https://javadao.xyz/forum.php?mod=viewthread&tid=39 ...
  • javaSE學習筆記(10) List、Set 1、數據存儲的數據結構 常見的數據結構 數據存儲的常用結構有:棧、隊列、數組、鏈表和紅黑樹。 1、棧 棧 : stack ,又稱堆棧,它是運算受限的線性表,其限制是僅允許在標的一端進行插入和刪除操作,不允許在其他任何位置進行添加、查找、刪除等操作。 簡 ...
  • 1. getById 的執行 前面一篇 提到過, Mapper.java 創建的時候, 會通過 jdk 代理的方式來創建, 且代理處理類為: MapperProxy . 所以當執行 UserMapper 的 getById 方法的時候, 就會去 MapperProxy 中執行 invoke 方法. ...
  • 一、前言 前面講了服務是如何導出到註冊中心的。其實Dubbo做的一件事就是將服務的URL發佈到註冊中心上。那現在我們聊一聊消費者一方如何從註冊中心訂閱服務併進行遠程調用的。 二、引用服務時序圖 首先總的來用文字說一遍內部的大致機制 Actor:可以當做我們的消費者。當我們使用@Reference註解 ...
  • 一. Mapper.java 創建過程 在前面註冊 bean 的時候, 對beanClass 進行了替換, 為 MapperFactoryBean. 那麼創建實例的時候, 會調用 MapperFactoryBean 的 getObject() 方法得到實例. @Override public T g ...
  • 最近在學著編寫一個操作系統的簡單內核,需要debug工具,我們這裡使用gdb來進行調試,由於虛擬機運行和本機是兩個部分,所以使用 gdb 的遠程調試技術,這裡對 gdb 常見調試以及遠程調試方式做一個總結。 遠程調試 先對在調試操作系統內核時用到的命令做一個說明(這裡省略了一部分makefile的內 ...
  • 舊版的mongo擴展已經不推薦使用了,在php7以上一般是安裝和使用新版的mongodb擴展 ubuntu下 apt-get install php-mongodb 例如下麵的代碼進行了查詢和插入集合操作 <?php class DocModel{ public $mongoManger=null; ...
  • 去除內嵌tomcat和添加jsp依賴 去除內嵌tomcat 在springboot啟動依賴中去除內嵌tomcat org.springframework.boot spring-boot-starter-web ... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...