当前位置: 首页 > article >正文

C++ 语言特性21 - 别名模板

一:概述

        别名模板是 C++11 引入的,用于为一个模板类型定义别名,从而简化复杂的模板类型定义。它结合了 using 关键字,可以对模板类型进行重新命名,使代码更加简洁和可读。

        1. 作用

  • 定义模板类型的别名
  • 简化复杂的模板类型,如容器、指针等。

        2. 语法

template<typename T>
using AliasName = ExistingType<T>;

//例子
#include <vector>

// 将 std::vector<T> 别名为 Vec<T>
template<typename T>
using Vec = std::vector<T>;

int main() {
    Vec<int> v = {1, 2, 3};  // 实际上是 std::vector<int>
}

//Vec<int> 是 std::vector<int> 的别名,通过别名模板,代码变得更加简洁。

二:使用场景 

        1. 简化复杂类型:如果某个模板类型嵌套得非常深,用别名模板可以减少冗长的代码

template<typename T>
using MapType = std::map<std::string, std::vector<T>>;

         2. 自定义智能指针:使用别名模板可以为智能指针(如 std::shared_ptrstd::unique_ptr)创建易于使用的别名。

template<typename T>
using Ptr = std::shared_ptr<T>;

Ptr<int> p = std::make_shared<int>(42);

        3. 替代 typedef:别名模板可以完全替代传统的 typedef,尤其是在处理模板类型时。

typedef std::vector<int> VecInt;  // 传统的 typedef
using VecIntAlias = std::vector<int>;  // 别名模板(推荐方式)

三:与 typedef 的对比

   1. typdef 语法

      typedef 是 C++98 及更早版本中定义类型别名的方式。

typedef ExistingType NewType;


//例子
typedef int Integer;  // 将 int 定义为 Integer
typedef int* IntPtr;  // 将 int* 定义为 IntPtr

Integer a = 10;
IntPtr p = &a;

   2.typedef 特点

  • 简单类型别名typedef 可以为基本类型、指针、函数指针、数组等定义别名。
  • 语法复杂typedef 在面对复杂的类型声明时,语法显得比较繁琐且不直观。例如函数指针的 typedef 语法容易混淆。

   3.typedef 局限性

  • 不支持模板typedef 无法与模板配合使用,无法定义模板类型的别名。

    4. 使用建议

        在现代 C++ 中,推荐使用 using 来替代 typedef,因为它的语法更简洁明了,但在现代项目中,typedef 基本已被 using 所取代。

四:与 typename 的对比

        typename 是一个用于声明模板参数或在模板中指定嵌套依赖类型的关键字。它的主要作用是告诉编译器:某个标识符是一个类型而非变量或其他东西。typename 在模板编程中非常重要,用于区分嵌套依赖类型(dependent type)和其他非类型成员。

   1. typename 作用

  • 声明模板参数:作为类型声明的一部分,表明这是一个类型。
  • 解决依赖类型问题:当类型依赖于另一个模板参数时,typename 用于明确告知编译器该标识符是类型而非变量

   2. typename 语法

template<typename T>
class MyClass {
public:
    typename T::NestedType member;  // 声明嵌套类型
};
template<typename T>
class MyClass {
public:
    typename T::value_type getValue(T container) {
        return container[0];
    }
};

int main() {
    std::vector<int> vec = {1, 2, 3};
    MyClass<std::vector<int>> obj;
    int value = obj.getValue(vec);
}


//在这个例子中,T::value_type 是一个依赖类型,使用 typename 明确告诉编译器它是一个类型。

3. 何时需要使用 typename 

        当在模板类或函数中使用某个依赖类型时,必须用 typename 来告知编译器这个符号是类型而不是变量或函数名。例如

template<typename T>
void func() {
    typename T::value_type x;  // T::value_type 是依赖于模板参数 T 的类型
}

//如果省略 typename,编译器会认为 T::value_type 是一个静态成员,而非类型,从而导致编译错误。

 4. 使用场景

  • 使用别名模板:当你希望简化复杂的模板类型定义,或为模板类/函数提供更简洁的命名时,使用别名模板。
  • 使用 typename:当你在模板中遇到依赖类型时(尤其是在嵌套模板结构中),使用 typename 来帮助编译器正确解析这个符号为类型。
// 使用别名模板简化模板类型
template<typename T>
using Ptr = std::shared_ptr<T>;

// 使用 typename 解决依赖类型问题
template<typename T>
void func() {
    typename T::NestedType x;  // T::NestedType 是依赖类型
}

//总之,别名模板用于为模板类型定义别名,简化代码,而 typename 则用于在模板中声明某个标识符为类型,尤其是在处理依赖类型时显得尤为重要。


http://www.kler.cn/news/336986.html

相关文章:

  • Python 之进阶语法:with...as...
  • ubuntu下vscode插件arm keil studio pack遇到的问题
  • 京东web 京东e卡绑定 第二部分分析
  • OpenCV视频I/O(17)视频写入类VideoWriter之检查视频编写器是否已经成功初始化的函数isOpened()的使用
  • 【QT Quick】C++交互:与QML类型转换
  • macOS开发环境配置与应用开发问题的延伸探讨
  • 什么是安全运营中心 SOC?
  • ARM base instruction -- movk
  • 将自己写好的项目部署在自己的云服务器上
  • Raft 常见问题解答
  • 开源2+1链动S2B2C商城小程序下社区团长的社群温度营造与商业价值实现
  • STM32 通用同步/异步通信
  • 基于SpringBoot图书馆预约与占座小程序【附源码】
  • JS测试框架——Jest
  • selenium元素定位
  • 仿小米的Disucz模板
  • TypeScript 算法手册【快速排序】
  • CSP-S复习:图论题选讲
  • 【网络安全】基础知识详解(非常详细)零基础入门到精通
  • Ubuntu24 Firefox和Window Firefox同步问题