用现代 C++ 等效结构替换 C 可变参数

Tes*_*ser 1 c++ c++20

我正在尝试用 C++ 等效项重写旧的 C 代码。我正在开发的模块似乎大量使用 C 变量参数列表。常见用法示例之一如下(请注意,下面的代码只是指示性示例,它不是实际的生产代码。我的目的是用现代 C++ 结构重写 VARARGS 的用法):

#include <stdio.h>
#include <stdarg.h>
 
void simple_printf(const char* fmt, ...)
{
    va_list args;
    va_start(args, fmt);
 
    while (*fmt != '\0') {
        if (*fmt == 'd') {
            int i = va_arg(args, int);
            printf("%d\n", i);
        } else if (*fmt == 'c') {
            // A 'char' variable will be promoted to 'int'
            // A character literal in C is already 'int' by itself
            int c = va_arg(args, int);
            printf("%c\n", c);
        } else if (*fmt == 'f') {
            double d = va_arg(args, double);
            printf("%f\n", d);
        }
        ++fmt;
    }
 
    va_end(args);
}
 
int main(void)
{
    simple_printf("dcff", 3, 'a', 1.999, 42.5); 
}
Run Code Online (Sandbox Code Playgroud)

我有兴趣知道现代 C++ 中有哪些可用的替代方案来重写与上面类似的代码。我也有兴趣了解现代 C++ 中可用的多种选项。代码示例会更有益。

bol*_*lov 7

该函数唯一“特别”的事情是在每个参数后面添加一个换行符。在 c++17 中,您可以使用折叠表达式来做到这一点:

template<typename... Args>
void print(Args... args) {
    ((std::cout << args << '\n'), ...); 
}
Run Code Online (Sandbox Code Playgroud)

在旧版本中,您可以通过递归来完成:

template<typename T>
void print(T value) {
  std::cout << value << std::endl;
}

template<typename T, typename... Args>
void print(T value, Args... args) {
  print(value);
  print(args...);
}
Run Code Online (Sandbox Code Playgroud)

无论如何,这有点奇怪。现代 C++ 将使用格式。fmt库或C ++20 标准格式库

std::cout << std::format("{}\n{}\n{}\n{}\n", 3, 'a', 1.999, 42.5);
Run Code Online (Sandbox Code Playgroud)

在 C++23 中:

std::print("{}\n{}\n{}\n{}\n", 3, 'a', 1.999, 42.5);
Run Code Online (Sandbox Code Playgroud)