C++17引入的折叠表达式简化了变参模板的使用,通过一元或二元操作符直接作用于参数包,避免了传统递归写法的冗长与复杂,支持求和、打印、逻辑判断等场景,显著提升了代码可读性和编写效率。

C++17引入的折叠表达式(Fold Expressions)无疑是变参模板(Variadic Templates)的一大福音,它让处理参数包变得异常简洁和直观,彻底告别了过去那些冗长且容易出错的递归模板写法。说白了,它就是一种把二元操作符“折叠”到参数包所有元素上的语法糖,让代码更易读、更紧凑。
折叠表达式的核心思想,在于提供了一种直接对参数包(parameter pack)应用二元操作符的方式,而无需手动编写递归基和递归步。这简直是变参模板的“救星”。
举个最简单的例子,如果你想计算一堆数字的和:
template<typename... Args>
auto sum(Args... args) {
// 传统方式可能需要递归函数或者借助辅助类
// 但有了折叠表达式,一切都变了
return (args + ...); // 一元右折叠,等价于arg1 + (arg2 + (arg3 + ...))
}
// 调用:sum(1, 2, 3, 4) -> 10再比如,你想把所有参数打印出来:
立即学习“C++免费学习笔记(深入)”;
#include <iostream>
template<typename T, typename... Args>
void print_all(T first_arg, Args... rest_args) {
// 以前可能得写:
// std::cout << first_arg << " ";
// if constexpr (sizeof...(rest_args) > 0) {
// print_all(rest_args...);
// }
// 现在:
std::cout << first_arg;
((std::cout << " " << rest_args), ...); // 二元左折叠,逗号操作符
std::cout << std::endl;
}
// 调用:print_all(1, "hello", 3.14, true);
// 输出:1 hello 3.14 1这里的
((std::cout << " " << rest_args), ...)
(std::cout << " " << rest_args_1, (std::cout << " " << rest_args_2, (...)))
折叠表达式有四种形式:
(pack op ...)
(args + ...)
arg1 + (arg2 + (... + argN))
(... op pack)
(... + args)
((arg1 + arg2) + ...) + argN
(init op ... op pack)
(0 + ... + args)
0 + (arg1 + (arg2 + ... + argN))
(pack op ... op init)
(args + ... + 0)
((arg1 + arg2) + ...) + argN + 0
实际使用中,二元折叠(带初始值)和一元右折叠(对于某些操作符)是最常见的。
说实话,在C++17之前,处理变参模板简直是件体力活。每当你需要对参数包里的每个元素执行一个操作,比如求和、打印、调用成员函数,你都得写一个递归模板函数。这意味着你不仅要有一个处理单个元素的“递归基”,还要有一个处理剩余参数的“递归步”。代码量大不说,逻辑也显得有点绕,特别是对于初学者来说,理解起来更是费劲。
// 以前的递归求和
template<typename T>
T sum_old(T t) {
return t;
}
template<typename T, typename... Args>
T sum_old(T t, Args... args) {
return t + sum_old(args...);
}这种模式虽然强大,但重复性太高,而且在编译错误时,模板展开的错误信息往往让人头大。折叠表达式的引入,就像是把这些重复的递归模式抽象成了一个语言层面的特性。它解决了最核心的痛点:简化了变参模板的写法,减少了模板元编程的认知负担和代码量。 它让那些原本需要多行甚至多个函数才能完成的任务,现在只需一行代码就能搞定。这不仅仅是语法上的简洁,更是思维上的解放,让开发者可以更专注于业务逻辑,而不是模板展开的细节。它让C++在处理可变参数列表时,有了Python、JavaScript等语言的简洁性,同时保留了C++的性能优势。
折叠表达式支持C++中的所有二元操作符,包括算术运算符 (
+
-
*
/
%
&
|
^
<<
>>
&&
||
==
!=
<
>
<=
>=
,
.*
->*
常见的应用场景真的很多:
聚合操作:
(args + ...)
(1 * ... * args)
(args && ...)
(args || ...)
template<typename... Bools>
bool all_true(Bools... b) {
return (b && ...); // 只有当所有b都为true时才返回true
}
// all_true(true, false, true) -> false函数调用/方法链:
struct Worker {
void do_work() { std::cout << "Working...\n"; }
};template<typename... Workers> void make_them_work(Workers&... ws) { (ws.do_work(), ...); // 依次调用每个Worker的do_work方法 } // Worker w1, w2; make_them_work(w1, w2); // 输出:Working... Working...
容器初始化/构造函数转发:
template<typename... Args>
struct MyVector : std::vector<int> {
MyVector(Args... args) : std::vector<int>({args...}) { // C++11列表初始化
// 或者,如果需要更复杂的处理,例如将参数逐个添加到容器
// (this->push_back(args), ...); // 这在构造函数体里可能更常见
}
};
// MyVector v(1, 2, 3, 4);自定义流操作:
print_all
template<typename... Args>
std::ostream& operator<<(std::ostream& os, const std::tuple<Args...>& t) {
os << "(";
std::apply([&os](const Args&... args) {
((os << args << " "), ...); // 打印每个元素,后面跟空格
}, t);
os << ")";
return os;
}
// std::tuple<int, double, std::string> my_tuple = {1, 2.5, "hello"};
// std::cout << my_tuple; // 输出:(1 2.5 hello )这里结合了
std::apply
虽然折叠表达式强大且简洁,但它也不是完全没有“脾气”。有些细节,第一次用的时候可能就会让你摸不着头脑。
(init op ... op pack)
(pack op ... op init)
pack
init
(0 + ... + args)
args
0
(pack op ...)
(... op pack)
pack
&&
true
||
false
,
void()
+
*
/
(f(args) + ...)
(f(args) ... +)
,
&&
||
总的来说,折叠表达式是现代C++中处理变参模板的利器,它让代码更简洁、更易读。但就像任何强大的工具一样,理解其工作原理和潜在的“陷阱”是高效使用的关键。一旦你掌握了它,你会发现变参模板的编写体验将得到质的飞跃。
以上就是C++折叠表达式 变参模板简化技巧的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号