C++-可变参数与可变参数模板

可变参数与可变参数模板

可变参数

三个点...表示可变参数列表,必须放在函数所有参数的最后面。函数至少要有一个固定参数(即使这个参数只是占位作用),否则无法获取可变参数列表中的每个参数。C++中...前的逗号是可选的,C是强制的。
作为可变参数传递时,可能会发生类型提升,如char类型转化为int,float类型转化为double,类型提升详见这里这里
通过<stdarg.h>中的几个宏可以获取可变参数列表中的每个参数:

  • va_start。允许访问可变参数列表
  • va_arg。访问下一个可变参数列表中下一个参数
  • va_end。结束可变参数列表遍历
  • va_list。保存 va_start、va_arg、va_end 和 va_copy 所需的信息
    示例代码:
#include <iostream>
#include <cstdarg>
 
void simple_printf(const char* fmt,...) // C-style "const char* fmt, ..." is also valid
{
    va_list args;
    va_start(args, fmt);
 
    while (*fmt != '\0') {
        if (*fmt == 'd') {
            int i = va_arg(args, int);
            std::cout << i << '\n';
        } else if (*fmt == 'c') {
            // note automatic conversion to integral type
            int c = va_arg(args, int);
            std::cout << static_cast<char>(c) << '\n';
        } else if (*fmt == 'f') {
            double d = va_arg(args, double);
            std::cout << d << '\n';
        }
        ++fmt;
    }
 
    va_end(args);
}
 
int main()
{
    simple_printf("dcff", 3, 'a', 1.999, 42.5); 
}

输出:

3
a
1.999
42.5

注意,并非所有类型参数都可以通过宏这种方式来获取。

可变参数模板

这里先引入几个概念:

  • 模板参数包(template parameter pack)是接受零个或多个模板参数的模板参数
  • 函数参数包(function parameter pack)是接受零个或多个函数参数的函数参数
  • 具有至少一个参数包的模板称为可变参数模板
    可变参数模板通常是更好的选择,因为它既不会像上述可变参数有参数类型的限制,也不会发生类型提升,可变参数模板是类型安全的
    可变参数模板涉及的内容很多,这里我们只需要学会使用可变参数函数模板就可以了。
    类型模板参数定义语法:
typename|class ... pack-name(optional)

参数包展开语法:

pattern ...	

示例:

template<class... Types> void f(Types... args);
f();       // OK: args contains no arguments
f(1);      // OK: args contains one argument: int
f(2, 1.0); // OK: args contains two arguments: int and double

Types代表参数包名称,args代表参数包(?),参数包的展开如下:

template<class... Us> void f(Us... pargs) {}
template<class... Ts> void g(Ts... args)
{
    f(&args...); // “&args...” is a pack expansion
                 // “&args” is its pattern
}
g(1, 0.2, "a"); // Ts... args expand to int E1, double E2, const char* E3
                // &args... expands to &E1, &E2, &E3
                // Us... pargs expand to int* E1, double* E2, const char** E3

std::initializer_list

如果所有参数具有相同的类型,std::initializer_list是一个可选的方法,但是传入的参数均不能被修改,因为std::initializer_list只提供const指针来访问其元素。

参考:

  1. https://docs.microsoft.com/en-us/cpp/cpp/functions-with-variable-argument-lists-cpp?view=msvc-170
  2. https://en.cppreference.com/w/cpp/utility/variadic
  3. https://en.cppreference.com/w/cpp/language/variadic_arguments
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

mrbone11

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值