省略号...用于可变参数函数和模板,C风格需配合stdarg.h处理参数但不安全,C++11起推荐类型安全的可变参数模板,支持递归展开和折叠表达式,编译期处理无运行时开销。

在C++中,三个点 ...(称为省略号,ellipsis)有特定用途,主要用于可变参数函数和模板。它允许函数或模板接受数量不固定的参数。下面详细讲解它的两种主要应用场景。
1. 可变参数函数(Variadic Functions)
在C风格的可变参数函数中,... 用来表示函数可以接收不定数量和类型的参数。这类函数需要配合 stdarg.h(C语言头文件)中的宏来处理参数。
常见例子是 printf 函数:
#include#include
void printNumbers(int count, ...) {
va_list args;
va_start(args, count);
for (int i = 0; i
int val = va_arg(args, int);
printf("%d ", val);
}
va_end(args);
}
int main() {
printNumbers(4, 10, 20, 30, 40); // 输出:10 20 30 40
return 0;
}
说明:
- va_list 用于声明一个参数列表变量。
- va_start 初始化该变量,必须传入最后一个命名参数。
- va_arg 获取下一个参数,需指定类型。
- va_end 清理资源。
注意:这种写法不安全,编译器无法检查参数类型和数量,容易引发崩溃。
立即学习“C++免费学习笔记(深入)”;
2. 可变参数模板(Variadic Templates)
C++11 引入了可变参数模板,使用 ... 来定义能接受任意数量模板参数的函数或类模板。这是现代C++推荐的方式,类型安全且灵活。
示例:递归展开参数包
#include iostream>// 基础版本:无参数
void print() {
std::cout
}
// 可变参数模板版本
template
void print(T first, Types... rest) {
std::cout
print(rest...); // 递归调用
}
int main() {
print(1, 2.5, "hello", 'A'); // 输出:1 2.5 hello A
return 0;
}
说明:
- typename... Types 表示模板参数包。
- Types... rest 是函数参数包。
- rest... 将参数包展开并传递。
还可以用折叠表达式(C++17)简化:
templatevoid print(Types... args) {
(std::cout }
3. 使用注意事项
安全性:
- C风格可变参数函数不进行类型检查,易出错。
- 可变参数模板在编译期展开,类型安全,推荐优先使用。
参数包的处理方式:
- 可通过递归、初始化列表、lambda等手段展开参数包。
- 注意避免无限递归,应提供终止特化或重载。
性能:
- 模板方式在编译期处理,运行时无额外开销。
- 参数包展开可能生成多个函数实例,增加代码体积。
基本上就这些。省略号在C++中不是注释,而是实现“接受任意参数”的关键语法。现代C++中,优先使用可变参数模板替代传统的 ... 函数,更安全也更强大。











