C++20中的预处理器宏——__VA_OPT__

一、预处理器宏

预处理,Preprocessing,它是在正式编译前对源代码的一种文本层面的处理。主要包括宏展开、文件包含、条件编译以及注释等的处理。预处理器则是执行上面的预处理的一种工具,它会不涉及到语法或语义的操作。预处理器宏就是宏,与普通的宏没有本质区别。不过,它是用于进行预处理的。常见的预处理器宏有:#ifdef、#ifndef、#if等以及一些特定的功能宏如_FILE_ 和 __LINE__等。

二、_VA_OPT _ 和 VA_ARGS

在C语言的printf函数的实现中,大家都接触过__VA_ARGS__这个宏。这个宏在应用时有不少的小细节需要处理。其中,##_VA_ARGS__即VA_ARGS__前面加上"##"可以处理当可变参数个数为0的时候,去掉前面多余的逗号(","),防止编译器报错。不过这个是GNU编译器的扩展,不属于标准库中的定义。可以参看看下面的示例代码。
而在C++20中为标准化统一,提供了"_VA_OPT
"这个宏用来处理空参数的问题。

c 复制代码
#define PRINT(...) printf(__VA_ARGS__)
#define PRINT_FMT(format, ...) printf(format, __VA_ARGS__)
#define PRINT_GNU(format, ...) printf(format, ##__VA_ARGS__)
#define PRINT_CPP20(format, ...) printf(format __VA_OPT__(, ) __VA_ARGS__)
int main() {
  PRINT("test macro print\n");
  // PRINT_FMT("test macro print fmt\n");//空参数
  PRINT_FMT("test macro print fmt %d\n", 100);
  PRINT_GNU("test macro print gnu\n");
  PRINT_GNU("test %s", "my print\n");
  return 0;
}

三、分析说明

在上面也看到了__VA_ARGS__和__VA_OPT__的应用,__VA_OPT__应用的方法是:

c 复制代码
__VA_OPT__(content)

对其来说当 VA_ARGS__展开非零参数时,插入"展开的参数";当*VA_ARGS* 展开为空(0参数)时,忽略","或其它引起错误的情况。虽然__VA_OPT__是C++20标准提出的,但要使用还是需要看编译器的具体支持。其优势在于处理老式的可变参数宏处理时的漏洞,从而有条件的处理不同情况下的宏展开。特别是在与兼容C语言开发时,有着重要的作用。

不过,对于C++来说,如果不想使用__VA_OPT__可以使用变参模板和参数包展开等方式来来替代,这就看实际应用的具体情况了。

四、应用

宏的应用还是比较多的,只是不被推荐罢了。__VA_OPT__可以应用于条件处理、复杂数据的创建以及变参模板的支持等。看下面例程:

c 复制代码
#include <iostream>
#include <tuple>
// tuple create
#define CREATE_TUPLE(...) std::make_tuple(__VA_OPT__(__VA_ARGS__))

//   create a named tuple
#define NAME_TUPLE(name, ...) std::tuple_cat(std::make_tuple(name) __VA_OPT__(, std::make_tuple(__VA_ARGS__)))

void test() {
  auto a = CREATE_TUPLE();
  auto b = CREATE_TUPLE(1, 'a', 2.0, "abc");

  auto c = NAME_TUPLE("single");
  auto d = NAME_TUPLE("two", 1, 1.1);
  auto e = NAME_TUPLE();

  std::cout << std::tuple_size<decltype(a)>::value << std::endl;
  std::cout << std::tuple_size<decltype(b)>::value << std::endl;
  std::cout << std::tuple_size<decltype(c)>::value << std::endl;
  std::cout << std::tuple_size<decltype(d)>::value << std::endl;
  std::cout << std::tuple_size<decltype(e)>::value << std::endl;
  std::cout << std::get<1>(d) << std::endl;
  std::cout << std::get<0>(c) << std::endl;
}
int main() {
  test();
  return 0;
}

五、总结

虽然说宏的应用场景在不断的被压缩,但在某些场景下,还是无法替代的。特别是在预处理时,宏的应用非常广泛。所以C++20提供__VA_OPT__用于解决__VA_ARGS__的空参数的特殊场景也是可以理解的。这也可以明白为什么C/C++标准迭代的复杂性,历史既是功绩也是包袱。

相关推荐
Codeking__3 天前
C++20的consteval和constinit(接C++11的constexpr)
算法·c++20
六bring个六5 天前
C++20协程
c++20·协程
C++实习生5 天前
Visual C++ 2005 Express 中文版
express·c++20
Ethan Wilson7 天前
VS2019 C++20 模块相关 C1001: 内部编译器错误
开发语言·c++·c++20
DYS_房东的猫8 天前
《 C++ 零基础入门教程》第10章:C++20 核心特性 —— 编写更现代、更优雅的 C++
java·c++·c++20
ice_junjun18 天前
C++20 线程返回值处理指南
c++20·c++ 多线程返回值
凌乱风雨121121 天前
从源码角度解析C++20新特性如何简化线程超时取消
前端·算法·c++20
shuai132_1 个月前
【无标题】
c++20
ULTRA??1 个月前
基于range的函数式编程C++,python比较
c++·python·kotlin·c++20