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

原著阅读地址

相关推荐
端平入洛19 小时前
delete又未完全delete
c++
端平入洛2 天前
auto有时不auto
c++
哇哈哈20213 天前
信号量和信号
linux·c++
多恩Stone3 天前
【C++入门扫盲1】C++ 与 Python:类型、编译器/解释器与 CPU 的关系
开发语言·c++·人工智能·python·算法·3d·aigc
蜡笔小马3 天前
21.Boost.Geometry disjoint、distance、envelope、equals、expand和for_each算法接口详解
c++·算法·boost
超级大福宝3 天前
N皇后问题:经典回溯算法的一些分析
数据结构·c++·算法·leetcode
weiabc3 天前
printf(“%lf“, ys) 和 cout << ys 输出的浮点数格式存在细微差异
数据结构·c++·算法
问好眼3 天前
《算法竞赛进阶指南》0x01 位运算-3.64位整数乘法
c++·算法·位运算·信息学奥赛
yyjtx3 天前
DHU上机打卡D31
开发语言·c++·算法
czxyvX3 天前
020-C++之unordered容器
数据结构·c++