【Effective Modern C++】第三章 转向现代C++:9. 优先选用别名声明,而非typedef

具有两种声明别名的方法:

c++ 复制代码
typedef std::unique_ptr<std::unordered_map<std::string, std::string>> UPtrMapSS; 

using UPtrMapSS = std::unique_ptr<std::unordered_map<std::string, std::string>>;

别名声明在处理涉及函数指针的类型时比较容易理解:

c++ 复制代码
//FP是一个指向函数的指针的同义词,它指向的函数带有int和const std::string&形参,不返回任何东西
typedef void (*FP)(int, const std::string&);    //typedef

//含义同上
using FP = void (*)(int, const std::string&);   //别名声明

别名声明可以模板化(被称为别名模板),而typedef不行:

使用别名声明:

c++ 复制代码
template<typename T> 
using MyAllocList = std::list<T,MyAlloc<T>>; // MyAllocList<T>是std::list<T,MyAlloc<T>>的同义词

MyAllocList<Widget> lw; // 客户代码

使用typedef需要从头开始:

c++ 复制代码
template<typename T>                            
struct MyAllocList {                           
    typedef std::list<T, MyAlloc<T>> type;      
}; //MyAllocList<T>是std::list<T, MyAlloc<T>>的同义词

MyAllocList<Widget>::type lw; // 客户代码

模板内使用的差异:typename 的必要性

模板内引用嵌套typedef的类型时,必须加typename前缀(因MyAllocList<T>::type依赖类型 ------ 编译器无法确定特化版本中::type是否为类型):

c++ 复制代码
template<typename T>
class Widget {
private:
    // 必须加typename:MyAllocList<T>::type依赖于T,可能被特化为非类型(如数据成员)
    typename MyAllocList<T>::type list; 
};

// 特化示例:MyAllocList<Wine>::type不是类型,而是数据成员
class Wine {};
template<>
class MyAllocList<Wine> {
private:
    enum class WineType { White, Red, Rose };
    WineType type; // 此处type是数据成员,非类型别名
};

而别名模板无需typename(编译器明确其结果必为类型,属于非依赖类型):

c++ 复制代码
template<typename T>
class Widget {
private:
    MyAllocList<T> list; // 无typename、无::type,简洁且无歧义
};

类型特性(type traits)场景的适配

C++11 的 type traits(如类型转换)基于struct嵌套typedef实现,使用时需::type后缀 + 模板内typename

c++ 复制代码
// C++11:需::type后缀,模板内还需加typename
std::remove_const<T>::type;          // 移除const修饰
std::remove_reference<T>::type;      // 移除引用修饰
std::add_lvalue_reference<T>::type;  // 添加左值引用

C++14 提供了基于别名声明的简化版本(后缀_t),无需::typetypename;即便仅支持 C++11,也可手动仿写这些别名模板:

c++ 复制代码
// C++14简化版
std::remove_const_t<T>;              // 等价于std::remove_const<T>::type
std::remove_reference_t<T>;          // 等价于std::remove_reference<T>::type

// C++11手动仿写别名模板(兼容方案)
template <class T> 
using remove_const_t = typename std::remove_const<T>::type;

总结

  • typedef不支持模板化,但别名声明支持。
  • 别名模板可以让人免写::type后缀,并且在模板内,对于内嵌typedef的引用经常要求加上typename前缀。

原著在线阅读地址

相关推荐
zwenqiyu20 小时前
P5283 [十二省联考 2019] 异或粽子题解
c++·学习·算法
Queenie_Charlie20 小时前
哈夫曼树
数据结构·c++·哈夫曼树
lihao lihao20 小时前
Linux信号
开发语言·c++·算法
大白话_NOI21 小时前
【洛谷 P2249】查找(深基 13. 例 1)+ 详细分析
c++·算法
智者知已应修善业21 小时前
【51单片机2个外部中断显示中断历时,初始化8左移3位共阳数码管】2024-6-6
c++·经验分享·笔记·算法·51单片机
码之气三段.21 小时前
edu158-B
c++
青梅橘子皮21 小时前
Linux---进程控制(2)(进程程序替换)
linux·c++·算法
chase_my_dream1 天前
A-LOAM中scanRegistration.cpp详细讲解
c++·人工智能·自动驾驶
王老师青少年编程1 天前
2022年CSP-X复赛真题及题解(T1:独木桥)
c++·真题·csp·信奥赛·复赛·独木桥·csp-x
John_ToDebug1 天前
Chromium 132→148 升级实战:Legacy IPC 消息丢失问题深度解析
c++·chrome·ai·架构