c++ 中的可变参数模板与折叠表达式

c++ 11 引入了可变参数模板,c++ 17 引入了折叠表达式,比 c 语言的可变参数更加简洁灵活。这篇博客总结了一些例子。

  • ...(省略号)用于可变参数(Variadic Arguments),它可以放在模板参数函数参数的后面,表示该参数是参数包(Parameter Pack),可以接收多个值。

1. 可变参数模板

递归调用是处理可变参数模板的经典方式,即通过递归地拆解参数包。

cpp 复制代码
#include <iostream>

// 基本函数(终止递归)
void print() {
    std::cout << "结束" << std::endl;
}

// 可变参数模板
template <typename T, typename... Args> // 这个... 表示多种不同的变量类型
void print(T first, Args... rest) { // 这个... 表示多种不同的变量类型
    std::cout << first << " ";  // 先打印当前参数
    print(rest...);             // 递归调用,传入剩余参数, 这个... 表示多种不同的传入参数
}

int main() {
    print(1, 2.5, "Hello", 'A');  
    // 输出: 1 2.5 Hello A 结束
}
  • 需要有递归的终止函数,比折叠表达式麻烦点

  • 可变参数模板不仅可以用于函数,还可以用于 类模板。

cpp 复制代码
#include <iostream>

// 可变参数类模板
template <typename... Args>
class MyClass {
public:
    MyClass() {
        std::cout << "MyClass with " << sizeof...(Args) << " template parameters\n";
    }
};

int main() {
    MyClass<int, double, char> obj1;  // 输出: MyClass with 3 template parameters
    MyClass<> obj2;                   // 输出: MyClass with 0 template parameters
}
  • sizeof...(Args) 为一种特殊的语法,用来 获取模板参数包中的参数个数。

2. 折叠表达式

C++17 引入了 折叠表达式(Fold Expressions),可以将参数包展开,并应用一个二元操作符来对这些参数进行"折叠",将它们合并成一个单一的值。

折叠表达式有两种基本形式:

  • 左折叠(left fold), 操作从 最左侧 开始,依次将参数包中的元素与左侧的结果结合,语法:
cpp 复制代码
(... op args)
  • 右折叠(right fold),操作从 最右侧 开始,依次将参数包中的元素与右侧的结果结合,语法:
cpp 复制代码
(op args ...)

其中,op 表示运算符,可以是 +, -, *, /, &&, ||, << 等。可以看出,省略号在运算符左边就是左折叠,在运算符右边就是右折叠。

cpp 复制代码
#include <iostream>

template <typename... Args> auto subtractLeft(Args... args) {
  return (... - args); // 左折叠,依次进行减法
}

template <typename... Args> auto subtractRight(Args... args) {
  return (args - ...); // 右折叠,依次进行减法
}

int main() {
  std::cout << subtractLeft(10, 3, 2, 1) << std::endl; // 输出是 4
  std::cout << subtractRight(10, 3, 2, 1) << std::endl; // 输出是 8

  return 0;
}

第一个函数左折叠表示 10-3-2-1=4,第二个函数右折叠表示 10-(3-(2-1))=8.

  • 可以直接对可变参数包进行操作,避免递归展开。
cpp 复制代码
#include <iostream>

// 使用 C++17 折叠表达式
template <typename... Args>
void print(Args... args) {
    ((std::cout << args << " "), ...);
    std::cout << std::endl;
}

int main() {
    print(1, 2.5, "Hello", 'A');
    // 输出: 1 2.5 Hello A
}

对所有参数包求和:

cpp 复制代码
#include <iostream>

template <typename... Args>
auto sum(Args... args) {
    return (args + ...); // 左折叠,将所有参数求和
}

int main() {
    std::cout << sum(1, 2, 3, 4) << std::endl; // 输出: 10
}
相关推荐
野生的程序媛8 分钟前
重生之我在学Vue--第13天 Vue 3 单元测试实战指南
前端·javascript·vue.js·单元测试
Aphasia31114 分钟前
简单介绍清除浮动解决高度塌陷的四种方法✍🏻
前端·css
..过云雨25 分钟前
09.【C++】list链表(STL中的列表容器,C++封装的带头双向链表,可实现指定类型的增删查改,迭代器操作等功能)
开发语言·c++
阿巴~阿巴~26 分钟前
C/C++蓝桥杯算法真题打卡(Day6)
c语言·c++·蓝桥杯·动态规划
幼安22928 分钟前
日志统计(C++,模拟,双指针)
开发语言·c++·spring
Captaincc1 小时前
这款堪称编程界的“自动驾驶”利器,集开发、调试、提 PR、联调、部署于一体
前端·ai 编程
我是小七呦1 小时前
万字血书!TypeScript 完全指南
前端·typescript
simple丶1 小时前
Webpack 基础配置与懒加载
前端·架构
simple丶1 小时前
领域模型 模板引擎 dashboard应用列表及配置接口实现
前端·架构
冰夏之夜影1 小时前
【css酷炫效果】纯css实现液体按钮效果
前端·css·tensorflow