詳解C++17中nodiscard標記符的使用
前言
在C++ 17中引入了一個標記符nodiscard,用于聲明一個 “非棄值(no-discard)表達式”。那么在開始之前,我們需要了解一下什么是棄值表達式。
棄值表達式
棄值表達式,就是放棄獲取返回值的表達式。首先棄值表達式的返回值是非void類型的。一般,我們使用的棄值表達式,其返回值只是起次要的作用,而其本身的作用占主要。比如++i;就是一個棄值表達式,它的主要作用就是累加,但同時我們也可以選擇獲取其累加的返回值,只不過這是次要的。
再比如,C標準庫的文件寫入函數(shù),其聲明如下:
int __cdecl fputs(const char * __restrict__ _Str,FILE * __restrict__ _File);
它有一個int類型的返回值,用于獲取寫入狀態(tài),它的主要作用是寫入文件,我可以選擇不獲取狀態(tài),也可以選擇獲取狀態(tài):
fputs("Hello World",pFile);
int result = fputs("Hello World",pFile);
nodiscard標記符
那么我如果想向用戶建議獲取返回值,這時候,我就可以使用nodiscard標記符。它一般用于標記函數(shù)的返回值或者某個類。聲明語法為:
/* @since C++17 */
[[nodiscard]] return_type function();
/* @since C++20 */
[[nodiscard("message")]] return_type function();
/* Standard lib defination */
/*
#if __cplusplus >= 201703L
# define _GLIBCXX_NODISCARD [[__nodiscard__]]
#else
# define _GLIBCXX_NODISCARD
#endif
*/
_GLIBCXX_NODISCARD return_type function();
如果一個被nodiscard標記了的表達式,如果我們在使用時棄值了,而且沒有使用static_cast<void>將其轉(zhuǎn)化為void時,編譯器會拋出warning來提醒用戶獲取返回值。
函數(shù)非棄值聲明
[[nodiscard]] int func1(){
return 1;
}
[[nodiscard("nodiscared function")]] int func2(){
return 1;
}
int main(){
func1(); //warning C++17
func2(); //warning c++20
int a = func1(); //no warning
static_cast<void>(func1()); //no warning
}
結(jié)果如下:

類/枚舉類/結(jié)構(gòu) 非棄值聲明
class [[nodiscard]] A{};
enum class [[nodiscard]] B{X,Y};
struct [[nodiscard]] C{};
A createA(){
return A();
}
B createB(){
return B::X;
}
C createC(){
return C();
}
int main(){
createA();
createB();
createC();
}
輸出如下:
6.cpp: In function 'int main()':
6.cpp:22:12: warning: ignoring returned value of type 'A', declared with attribute 'nodiscard' [-Wunused-result]
22 | createA();
| ~~~~~~~^~
6.cpp:10:3: note: in call to 'A createA()', declared here
10 | A createA(){
| ^~~~~~~
6.cpp:6:21: note: 'A' declared here
6 | class [[nodiscard]] A{};
| ^
6.cpp:23:12: warning: ignoring returned value of type 'B', declared with attribute 'nodiscard' [-Wunused-result]
23 | createB();
| ~~~~~~~^~
6.cpp:14:3: note: in call to 'B createB()', declared here
14 | B createB(){
| ^~~~~~~
6.cpp:7:26: note: 'B' declared here
7 | enum class [[nodiscard]] B{X,Y};
| ^
6.cpp:24:12: warning: ignoring returned value of type 'C', declared with attribute 'nodiscard' [-Wunused-result]
24 | createC();
| ~~~~~~~^~
6.cpp:18:3: note: in call to 'C createC()', declared here
18 | C createC(){
| ^~~~~~~
6.cpp:8:22: note: 'C' declared here
8 | struct [[nodiscard]] C{};
| ^
返回類引用與類指針
當返回值為引用或者指針的 類/枚舉類/結(jié)構(gòu)(函數(shù)不行) 時,nodiscard 就無效了:
class [[nodiscard]] A{};
A& createAref(){
A* a = new A();
return *a;
}
A* createAptr(){
A* a = new A();
return a;
}
int main(){
createAref(); //no warning
createAptr(); //no warning
}
到此這篇關于詳解C++17中nodiscard標記符的使用的文章就介紹到這了,更多相關C++17 nodiscard標記符內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
C/C++ 中怎樣使用SetConsoleTextAttribute()函數(shù)來控制輸出字符的顏色
這篇文章主要介紹了C/C++ 中如何使用SetConsoleTextAttribute()函數(shù)來控制輸出字符的顏色,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03
淺談C語言中include""與include<>的區(qū)別
C語言中包含文件有兩種包含符號,一個是<>尖括號,另一個是""雙引號。那么這兩個有什么區(qū)別呢?本文就詳細的介紹一下,感興趣的可以了解一下2021-06-06
Qt中const?QString轉(zhuǎn)換?char?*可能的坑
本文主要介紹了Qt中const?QString轉(zhuǎn)換?char?*可能的坑,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-07-07
詳解C++編程中多級派生時的構(gòu)造函數(shù)和訪問屬性
這篇文章主要介紹了詳解C++編程中多級派生時的構(gòu)造函數(shù)和訪問屬性,是C++入門學習中的基礎知識,需要的朋友可以參考下2015-09-09
基于C++的農(nóng)夫過河問題算法設計與實現(xiàn)方法
這篇文章主要介紹了基于C++的農(nóng)夫過河問題算法設計與實現(xiàn)方法,簡單描述了農(nóng)夫過河問題,并結(jié)合實例形式詳細分析了基于C++實現(xiàn)農(nóng)夫過河問題的相關算法實現(xiàn)步驟與操作技巧,需要的朋友可以參考下2017-09-09

