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++标准迭代的复杂性,历史既是功绩也是包袱。

相关推荐
普通网友6 天前
记录我适配iOS26遇到的一些问题
c++20
前进吧-程序员6 天前
C++20/23 Ranges:从「迭代器对」到「可组合管道」
c++20
Shan120511 天前
实例分析:C++20的std::jthread
c++20
charlie11451419111 天前
基于开源项目的现代C++工程实践——OnceCallback 前置知识(下):C++20/23 高级特性
c++·开源·c++20
Hical_W11 天前
Hical 踩坑实录五部曲(二):MSVC / GCC / Clang 三平台 C++20 编译差异
linux·windows·经验分享·嵌入式硬件·macos·开源·c++20
Shan120513 天前
C++20中带有约束条件的new
c++20
Hical_W16 天前
用 Hical + MySQL 5 分钟搭建 CRUD API(C++20 协程版)
数据库·mysql·c++20
Hical_W16 天前
从 io_context 出发,掌握 C++20 协程式异步 I/O,学会 TCP 服务器、定时器和多线程模型,结合 Hical 框架实战解读
服务器·tcp/ip·开源·c++20
c++之路20 天前
C++20概述
java·开发语言·c++20
故事还在继续吗21 天前
C++20关键特性
开发语言·c++·c++20