当前位置: 首页 > news >正文

C++17 nodiscard标记符

文章目录

  • 前言
  • 弃值表达式
  • nodiscard标记符
    • 函数非弃值声明
    • 类/枚举类/结构 非弃值声明
    • 返回类引用与类指针

前言

在C++ 17中引入了一个标记符nodiscard,用于声明一个 “非弃值(no-discard)表达式”。那么在开始之前,我们需要了解一下什么是弃值表达式。

弃值表达式

弃值表达式,就是放弃获取返回值的表达式。首先弃值表达式的返回值是非void类型的。一般,我们使用的弃值表达式,其返回值只是起次要的作用,而其本身的作用占主要。比如++i;就是一个弃值表达式,它的主要作用就是累加,但同时我们也可以选择获取其累加的返回值,只不过这是次要的。

再比如,C标准库的文件写入函数,其声明如下:

int __cdecl fputs(const char * __restrict__ _Str,FILE * __restrict__ _File);

它有一个int类型的返回值,用于获取写入状态,它的主要作用是写入文件,我可以选择不获取状态,也可以选择获取状态:

fputs("Hello World",pFile);
int result = fputs("Hello World",pFile);

nodiscard标记符

那么我如果想向用户建议获取返回值,这时候,我就可以使用nodiscard标记符。它一般用于标记函数的返回值或者某个类。声明语法为:

/* @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>将其转化为void时,编译器会抛出warning来提醒用户获取返回值。

函数非弃值声明

[[nodiscard]] int func1(){return 1;
}[[nodiscard("nodiscared function")]] int func2(){return 1;
}int main(){func1();                         //warning C++17func2();                         //warning c++20int a = func1();                 //no warningstatic_cast<void>(func1());      //no warning
}

结果如下:
在这里插入图片描述

类/枚举类/结构 非弃值声明

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 here10 | A createA(){|   ^~~~~~~
6.cpp:6:21: note: 'A' declared here6 | 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 here14 | B createB(){|   ^~~~~~~
6.cpp:7:26: note: 'B' declared here7 | 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 here18 | C createC(){|   ^~~~~~~
6.cpp:8:22: note: 'C' declared here8 | struct [[nodiscard]] C{};|                      ^

返回类引用与类指针

当返回值为引用或者指针的 类/枚举类/结构(函数不行) 时,nodiscard 就无效了:

class [[nodiscard]] A{};A& createAref(){A* a = new A();return *a;
}A* createAptr(){A* a = new A();return a;
}int main(){createAref(); //no warningcreateAptr(); //no warning
}
http://www.lryc.cn/news/21701.html

相关文章:

  • SAP 寄售业务的标准流程
  • 操作系统高频知识
  • 加载预训练模型,模型微调,在自己的数据集上快速出效果
  • VScode远程连接服务器-过程试图写入的管道不存在-could not establist connection to【已解决】
  • 电子技术——B类输出阶
  • 【老卫搬砖】034期:HarmonyOS 3.1 Beta 1初体验,我在本地模拟器里面刷短视频
  • Day901.内部临时表 -MySQL实战
  • jstatd的启动方式与关闭方式
  • _improve-3
  • C++——异常
  • MVVM 架构进阶:MVI 架构详解
  • 有没有必要考PMP证书?
  • 1 机器学习基础
  • java基础系列(六) sleep()和wait() 区别
  • Urho3D序列化
  • 企业级信息系统开发学习1.3——利用注解配置取代Spring配置文件
  • VUE DIFF算法之快速DIFF
  • 一文掌握如何轻松稿定项目风险管理【静说】
  • 操作系统权限提升(十四)之绕过UAC提权-基于白名单AutoElevate绕过UAC提权
  • ecology9-谷歌浏览器下-pdf.js在渲染时部分发票丢失文字 问题定位及解决
  • JavaScript Window Navigator
  • Linux基础命令-du查看文件的大小
  • 文献计量分析方法:Citespace安装教程
  • MVI 架构更佳实践:支持 LiveData 属性监听
  • LeetCode438 找到字符串中所有字母异位词 带输入和输出
  • ACSC 2023 比赛复现
  • 【Linux驱动开发100问】什么是模块?如何编写和使用模块?
  • Android 9.0 Recent列表不显示某个app
  • 深度学习之卷积神经网络学习笔记一
  • 黑盒测试的常用方法