【Effective Modern C++】第二章 auto:6. 当auto推导的类型不符合要求时,使用显式类型初始化习惯用法

个人认为原著写的非常难懂,所以精简总结如下:

auto与代理类的问题:

当使用auto进行类型推导时,如果表达式返回的是代理类,auto会推导出代理类型而不是被代理的实际类型,可能导致未定义行为。

例如:

c++ 复制代码
std::vector<bool> features(const Widget& w);
bool highPriority = features(w)[5];  // 含义是w具有高优先级吗?
processWidget(w, highPriority);      // 按照w的优先级来处理

这段代码没有什么问题,但是如果把highPriority从显式类型改成auto

c++ 复制代码
auto highPriority = features(w)[5];  // 危险!推导出std::vector<bool>::reference
processWidget(w, highPriority);      // 可能产生未定义行为!

原因:

  1. std::vector<bool>内部使用紧凑结构存储(每个bool占1位)
  2. 内存最小寻址单位是字节,单个 "位" 没有独立的内存地址;
  3. operator[]无法返回bool&(不能对单个位取引用),因此返回一个代理类std::vector<bool>::reference
  4. auto不会自动把代理类转换成bool,而是直接推导成std::vector<bool>::reference类型。

因此需要显式声明 其为bool类型:

c++ 复制代码
auto highPriority = static_cast<bool>(features(w)[5]);  // 安全!

如何识别出代理类

头函数或函数签名会反映出其存在:

c++ 复制代码
namespace std {
    template<class Allocator>
    class vector<bool, Allocator> {
    public:
        class reference { ... };  // 代理类!
        reference operator[](size_type n);
        ...
    };
}

非同寻常的返回值类型:如:std::vector<T>::operator[]通常返回T&,如果返回类型不是T&,可能是代理类。

常见的代理类场景:

  • std::vector<bool>的相关操作:位无独立地址,无法返回引用。
  • 表达式模板(用于优化数值计算):临时对象多,计算效率低。
  • 某些智能指针的实现:内存访问不安全/权限难控。
  • 延迟求值(lazy evaluation)系统:提前计算可能无效。

其他应用场景

1. 精度控制

c++ 复制代码
double calcEpsilon();
auto ep = static_cast<float>(calcEpsilon());  // 明确表示精度降低

还可以应用于计算随机访问迭代器的容器中的某个元素下标。

2. 类型转换强调

c++ 复制代码
double d = 0.7;
size_t size = 100;
auto index = static_cast<int>(d * size);  // 明确表示浮点到整数的转换

3. 表达式模板的代理

c++ 复制代码
Matrix m1, m2, m3, m4;
auto sum = static_cast<Matrix>(m1 + m2 + m3 + m4);  // 避免代理类型

总结

  • "隐形"的代理类型可以导致auto根据初始化表达式推导出"错误的"类型。
  • 带显式类型的初始化习惯用法强制auto推导出你想要的类型。

原著阅读地址

相关推荐
tankeven18 小时前
C++ 学习杂记01:C++ vector 容器详细
c++
艾莉丝努力练剑18 小时前
【Linux线程】Linux系统多线程(八):<策略模式>日志系统的封装实现
linux·运维·服务器·c++·学习·策略模式
盐焗鹌鹑蛋18 小时前
【C++】string模拟实现
c++
特种加菲猫18 小时前
C++进阶:模板深度解析与继承机制初探
开发语言·c++
旖-旎18 小时前
递归(快速幂)(5)
c++·算法·力扣·递归
大江东去浪淘尽千古风流人物21 小时前
【cuVSLAM】GPU 加速、多相机、实时视觉/视觉惯性 SLAM设计优势
c++·人工智能·数码相机·ubuntu·计算机视觉·augmented reality
自信150413057591 天前
重生之从0开始学习c++之模板初级
c++·学习
历程里程碑1 天前
2. Git版本回退全攻略:轻松掌握代码时光机
大数据·c++·git·elasticsearch·搜索引擎·github·全文检索
极客智造1 天前
深度解析 C++ 类继承与多态:面向对象编程的核心
c++
零号全栈寒江独钓1 天前
基于c/c++实现linux/windows跨平台获取ntp网络时间戳
linux·c语言·c++·windows