std::tuple是编译期定长的异构数据聚合体,适用于封装逻辑相关、类型各异的小数据组(如多返回值、配置项);不可动态增删元素,须用std::get()或C++17结构化绑定访问,支持字典序比较。
它不是容器,而是编译期确定长度的异构数据聚合体。适合封装一组逻辑相关、生命周期一致、但类型各异的小数据(比如函数返回多个值、配置项组合、坐标+标签等)。别拿它当 std::vector 用——长度不能变,不支持运行时索引访问。
std::make_tuple 构造,用 std::get() 按索引取值构造必须显式或靠类型推导;取值必须用编译期常量索引(0、1),不能是变量。越界会编译失败,不是运行时错误。
std::get(t) 取第一个元素,索引从 0
开始std::get(t) 是错的——模板参数是整型非类型模板参数,不是类型std::get 返回的是引用,可直接修改原值auto t = std::make_tuple(42, "hello", 3.14); int a = std::get<0>(t); // OK const char* s = std::get<1>(t); // OK double d = std::get<2>(t); // OK // std::get<3>(t); // 编译错误:越界
C++17 起推荐用 auto [a, b, c] = t;,比手写一堆 std::get 更清晰、不易错序。注意:绑定名数量、顺序、类型必须和 tuple 元素完全一致,否则编译失败。
auto [a, , c] = t; 是非法的)_(但这是 C++20 起支持,且非所有编译器默认启用)auto t = std::make_tuple(100, std::string("test"), true);
auto [num, str, flag] = t; // C++17,推荐
// num 是 int,str 是 std::string(拷贝),flag 是 bool
str += " done"; // 修改的是拷贝,不影响原 tuple 中的 string返回 std::tuple 的函数常用于多值返回。调用时若用结构化绑定接收,编译器通常能优化掉拷贝(RVO 或移动),但前提是 tuple 内部成员支持移动(比如含 std::string 或 std::vector 就行)。
std::tuple f() { return std::make_tuple(a, b); } 然后在调用处反复 std::get——破坏可读性最易被忽略的一点:tuple 的比较操作符(==、)要求所有元素都支持对应操作,且按字典序逐个比较。别默认以为“能构造就能比较”。