【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推导出你想要的类型。

原著阅读地址

相关推荐
炸膛坦客5 小时前
单片机/C/C++八股:(二十)指针常量和常量指针
c语言·开发语言·c++
炸膛坦客7 小时前
单片机/C/C++八股:(十九)栈和堆的区别?
c语言·开发语言·c++
2401_831824967 小时前
代码性能剖析工具
开发语言·c++·算法
是wzoi的一名用户啊~8 小时前
【C++小游戏】2048
开发语言·c++
Sunshine for you8 小时前
C++中的职责链模式实战
开发语言·c++·算法
qq_416018729 小时前
C++中的状态模式
开发语言·c++·算法
2401_884563249 小时前
模板代码生成工具
开发语言·c++·算法
2401_831920749 小时前
C++代码国际化支持
开发语言·c++·算法
m0_672703319 小时前
上机练习第51天
数据结构·c++·算法
2401_851272999 小时前
自定义内存检测工具
开发语言·c++·算法