0

0

C++怎么用变长参数 C++11模板参数包展开方法【高级】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-03-01 13:48:11

|

580人浏览过

|

来源于php中文网

原创

c++11后应优先用模板参数包而非va_list,因类型安全且编译期检查;展开方式有递归、初始化列表(c++11)、折叠表达式(c++17);需注意引用折叠、移动语义和tuple封装场景。

c++怎么用变长参数 c++11模板参数包展开方法【高级】

变长参数函数怎么写,va_list 还能用吗?

C++11 之后,va_list 并没被废掉,但不推荐在新代码里用——它不类型安全,编译器没法检查参数个数和类型,一错就是运行时崩溃。比如传 std::stringprintf 风格的函数,轻则乱码,重则段错误。

现代 C++ 应该用模板参数包(parameter pack),配合 ... 展开。它在编译期做类型推导,出错直接报编译错误,更可靠。

  • 必须用 template<typename... args></typename...> 声明可变模板
  • 参数包本身不能直接使用,得通过展开(如递归、折叠表达式或初始化列表技巧)
  • 不支持部分特化参数包,但可以偏特化整个模板

怎么安全展开参数包?三种常用手法对比

最常用的是递归展开逗号折叠表达式(C++17 起),C++11 只能靠递归或初始化列表“骗”编译器执行。

  • 递归方式:定义一个终止重载(空参数版本),再写一个带至少一个参数的模板,每次处理头元素,把剩余参数包传给自身
  • 初始化列表技巧(C++11 兼容):{(func(args), void()), ...} 利用列表构造顺序保证求值,但要求 func 返回 void 或忽略返回值
  • 折叠表达式(C++17)更简洁:((std::cout ,但老项目别硬上

示例(C++11 兼容):

立即学习C++免费学习笔记(深入)”;

飞书知识问答
飞书知识问答

飞书平台推出的AI知识库管理和智能搜索工具

下载
template<typename T>
void print_one(const T& t) { std::cout << t << "\n"; }
<p>template<typename T, typename... Args>
void print(const T& t, const Args&... args) {
print_one(t);
print(args...); // 尾递归展开
}</p>

参数包展开时最容易崩在哪几个地方?

不是所有语法都能直接套 ...,稍不注意就编译失败:

  • sizeof...(Args) 没问题,但 sizeof(Args...) 是错的(少括号)
  • 不能在 if 条件里直接展开,比如 if (args...) 合法性取决于上下文,大概率报错
  • 引用折叠容易翻车:T&& 在模板中是万能引用,但展开后如果 Args 包含左值,可能变成 T&&&,触发引用折叠规则,实际变成 T& —— 这不是 bug,但常被误以为是类型丢失
  • 移动语义要小心:std::move(args)... 会把每个参数都转成右值,但如果原参数是左值,后续再用就悬空了

什么时候该用 std::tuple 而不是裸参数包?

参数包本质是编译期“未打包”的类型序列,没法存起来、传出去、延迟展开。一旦需要:

  • 把一堆参数暂存到某个对象里(比如延迟调用)
  • 按索引取某个参数(std::get(t)
  • 遍历所有参数并统一处理(std::apply

就得先收进 std::tuple<args...></args...>。比如实现一个通用的包装器:

template<typename F, typename... Args>
auto delay_call(F&& f, Args&&... args) {
    auto t = std::make_tuple(std::forward<Args>(args)...);
    return [f = std::forward<F>(f), t = std::move(t)]() mutable {
        return std::apply(f, std::move(t));
    };
}

参数包展开看着灵活,但真正要用稳,得清楚它只是“编译期占位符”,不是容器。想存、想查、想复用,绕不开 std::tuplestd::apply 这一套。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

890

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

836

2023.08.22

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

76

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

300

2023.11.28

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

185

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

125

2025.11.27

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

429

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

599

2023.08.10

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

6

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.4万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 19.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号