【Effective Modern C++】第一章 类型推导:2. 理解 auto 类型推导

推导方式和1.理解模板类型推导一致,下面讨论不同点:

在初始化时,采用auto声明变量相比于采用固定类型声明变量更具优势。下列声明都能通过编译,但是结果却并不相同:

c++ 复制代码
auto x1 = 27; // 类型是int,值是27
auto x2(27); // 同上
auto x3 = {27}; // 类型是std::initializer_list<int>,值是{27}
auto x4{27}; //同上 

当用于auto声明变量的初始化表达式是使用大括号时,推导所得类型就属于std::initializer_list<int>。如果类型推导失败(例如大括号里的值类型不一样),代码就不通过编译:

c++ 复制代码
auto x5 = {1,2,3.0}; // 错误,推导不出std::initializer_list<T>中的T

这里发生了两次类型推导:

  1. auto的使用:x5的类型需要推导,由于x5的初始表达式是用大括号,所以x5必须推导为一个std::initializer_list
  2. std::initializer_list<T>T的推导:std::initializer_list是个模板,需要根据某个类型T产生类型std::initializer_list<T>,所以T的类型需要被推导出来。

但是,对于普通模板而言,不会自动识别为std::initializer_list

c++ 复制代码
template<typename T> 
void f(T param);

f({ 11, 23, 9 }); // 编译错误!无法推导T的类型

又但是,如果指定模板中的paramstd::initializer_list<T>,则在T的类型未知的情况下,模板类型推导机制会推导出T应该的类型:

c++ 复制代码
template<typename T> 
void f(std::initializer_list<T> initlist); 

f({ 11, 23, 9 }); // 编译通过!

总结

  • 在一般情况下,auto 类型推导和模板类型推导是一模一样的,但是 auto 类型推导会假定用大括号括起的初始化表达式代表一个 std::initializer_list,但模板类型推导却不会。
  • 在函数返回值或 lambda 式的形参中使用 auto , 意思是使用模板类型推导而非 auto 类型推导。

原著在线阅读地址

相关推荐
星火开发设计4 小时前
变量与常量:C++ 中 const 关键字的正确使用姿势
开发语言·c++·学习·const·知识
hetao17338374 小时前
2026-01-14~15 hetao1733837 的刷题笔记
c++·笔记·算法
a***59264 小时前
C++跨平台开发:挑战与解决方案
开发语言·c++
hetao17338374 小时前
2026-01-12~01-13 hetao1733837 的刷题笔记
c++·笔记·算法
Yu_Lijing4 小时前
基于C++的《Head First设计模式》笔记——外观模式
c++·笔记·设计模式
CoderCodingNo4 小时前
【GESP】C++六级考试大纲知识点梳理, (5) 动态规划与背包问题
开发语言·c++·动态规划
情缘晓梦.4 小时前
C++ 类和对象(完)
开发语言·jvm·c++
qq_433554545 小时前
C++ 图论算法:强连通分量
c++·算法·图论
郝学胜-神的一滴5 小时前
Linux条件变量:线程同步的利器
linux·服务器·开发语言·c++·程序人生·软件工程
曼巴UE55 小时前
UE5 C++ 里创造 和 使用编辑GamePlayTag
c++·ue5·ue