news 2026/4/15 3:47:28

【Effective Modern C++】第一章 类型推导:1.理解模板类型推导

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【Effective Modern C++】第一章 类型推导:1.理解模板类型推导

函数模板形如:

template<typename T> void f(ParamType param);

一次调用形如:

f(expr); //以某表达式调用 f

在编译期,编译器会通过expr推导T的类型和ParamType的类型(通常会包含一些修饰,例如const或引用符号等限定)。

T的类型推导结果分以下三种情况讨论:

1.ParamType具有指针或引用型别,但不是万能引用

类型推导规则如下:

  1. expr具有引用类型,先将引用部分忽略。
  2. 然后,对expr的类型和ParamType的类型执行模式匹配,来决定T的类型。
举例

模式声明如下:

template<typename T> void f(T& param); // param 是个引用

声明下列变量:

int x = 27; // x 的类型是 int const int xx= x; // cx 的类型是 const int const int& rx = x; // rx 是 x 的类型为 const int 的引用

各次调用的推导结果如下:

f(x); // T 的类型是 int,param 的类型是 int& f(cx); // T 的类型是 const int,param 的类型是 const int& f(rx); // T 的类型是 const int,param 的类型是 const int&

右值和指针方式同理:

template<typename T> void f(T* param); // param 是个指针 int x = 27; const int *px = &x; // px 是指向 x 的指针,类型为 const int f(&x); // T 的类型是 int, param 的类型是 int* f(px); // T 的类型是 const int, param 的类型是 const int*

2.ParamType是万能引用

类型推导规则如下:

  1. 如果expr是个左值,TParamType都会被推导为左值引用。
  2. 如果expr是个右值,则应用“常规"(即情况 1 中的)规则。
举例
template<typename T> void f(T&& param); // param 是个万能引用 int x = 27; const int cx= x; const int& rx = x; f(x); // x 是个左值,所以 T 的类型是 int&, param 的类型也是 int& f(cx); // cx 是个左值,所以 T 的类型是 const int&, param 的类型也是 const int& f(rx); // rx 是个左值,所以 T 的类型是 const int&, param 的类型也是 const int& f(27); // 27 是个右值,所以 T 的类型是 int, param 的类型就成了 int&&

3.ParamType既非指针也非引用

即按值传递,无论传入的是什么,param都会是它的副本,是一个新对象:

  1. expr具有引用类型,则忽略其引用部分。
  2. 忽略expr的引用性之后,若expr是个const对象,也忽略。若其是个volatile对象,同样忽略。
举例
template<typename T> void f(T param); //param现在是按值传递 int x = 27; //同前 const int cx = x; //同前 const int& rx = x; //同前 f(x); // T和param的类型都是int f(cx); // T和param的类型都是int f(rx); // T和param的类型都是int

即使cxrx代表const值,param仍然不具有const的类型。param是个完全独立于cxrx的一个副本。从而cxrx不可修改并不能说明param是否可以修改。

数组实参

一种边缘情况,数组类型有别于指针类型,但是在多数语境下,数组会退化成首元素的指针。

C++ 中没有真正的“数组形参”,因为函数无法直接接收数组;当数组传递给按值形参的模板时,编译器会把形参里的数组声明直接当作指针处理。但是:

尽管函数无法声明真正的数组类型的形参,但是它们能够将形参声明为数组的引用。在这种情况下T的类型会被推导成实际的数组类型。这个类型中会包含数组尺寸。

利用声明数组引用能力,可以创造出一个模板用来推导数组中含有的元素个数:

// 以编译期常量形式返回数组尺寸 // (该数组形参未起名字,因为我们只关心共含有的元素个数) template<typename T, std::size_t N> constexpr std::size_t arraySize(T (&)[N]) noexcept { return N; }

将函数声明为constexpr,能够使得其返回值在编译期就可以使用。从而在声明一个数组时,指定其尺寸和另一数组相同,而后者的尺寸则从花括号初始化形式计算得出:

int keyVals[] = { 1, 3, 7, 9, 11, 22, 35 }; // keyVals 含有 7 个元素 int mappedVals [arraySize(keyVals)]; // mappedVals 被指定与之相同

函数实参

函数类型也同样会退化成函数指针,并且针对数组类型推导的一切都适用于函数及其向函数指针的退化。如下:

void someFunc(int, double); // someFunc 是个函数,其类型为 void(int, double) template<typename T> void f1(T param); // 在 f1 中,param 按值传递 template<typename T> void f2 (T& param); // 在 f2 中,param 按引用传递 f1(someFunc); // param 披推导为函数指针,具体类型是 void (*)(int, double) f2(someFunc); // param 披推导为函数引用,具体类型是 void (&)(int, double)

总结

  • 在模板类型推导时,有引用的实参会被视为无引用,它们的引用会被忽略
  • 对于通用引用的推导,左值实参会被特殊对待
  • 对于传值类型推导,const/volatile实参会被认为是non-const的和non-volatile
  • 在模板类型推导时,数组名或者函数名实参会退化为指针,除非它们被用于初始化引用

原著在线阅读地址

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/13 16:33:33

AI 应用开发的运营

AI 应用的运营已不再是简单的“客服推广”&#xff0c;而是演变成了以 数据回流&#xff08;Data Loop&#xff09; 和 模型持续演进 为核心的系统工程。以下是 AI 应用运营的四大核心模块&#xff1a;1. 模型效果运营AI 应用上线只是开始&#xff0c;由于用户输入的随机性和“…

作者头像 李华
网站建设 2026/4/3 3:05:16

工业级语义分割新范式|SAM3大模型镜像技术解析与应用

工业级语义分割新范式&#xff5c;SAM3大模型镜像技术解析与应用 1. 引言&#xff1a;从几何感知到语义认知的工业视觉跃迁 传统工业视觉检测长期依赖于监督学习框架&#xff0c;即通过大量标注数据训练专用模型以识别特定缺陷。这一模式在面对多品种、小批量&#xff08;Hig…

作者头像 李华
网站建设 2026/4/12 23:41:37

BERT智能填空服务安全加固:输入过滤与异常检测实战

BERT智能填空服务安全加固&#xff1a;输入过滤与异常检测实战 1. 引言 1.1 业务场景描述 随着自然语言处理技术的普及&#xff0c;基于 BERT 的中文语义填空服务在教育辅助、内容创作和智能客服等场景中展现出广泛应用价值。本镜像基于 google-bert/bert-base-chinese 模型…

作者头像 李华
网站建设 2026/4/4 13:02:57

YOLOv9部署前必读:官方代码库与镜像差异对比说明

YOLOv9部署前必读&#xff1a;官方代码库与镜像差异对比说明 在将YOLOv9应用于实际项目之前&#xff0c;了解其官方代码库与预构建镜像之间的差异至关重要。许多开发者在使用深度学习模型时倾向于选择预配置的镜像以节省环境搭建时间&#xff0c;但往往忽视了镜像可能带来的版…

作者头像 李华
网站建设 2026/4/12 15:00:26

万物识别模型调用避坑指南:Python路径配置实战详解

万物识别模型调用避坑指南&#xff1a;Python路径配置实战详解 在当前AI应用快速落地的背景下&#xff0c;图像识别技术已成为智能系统的核心能力之一。阿里开源的“万物识别-中文-通用领域”模型凭借其对中文标签的良好支持和广泛的物体覆盖能力&#xff0c;正在被越来越多开…

作者头像 李华
网站建设 2026/4/2 21:18:14

Fun-ASR-MLT-Nano-2512性能优化:批量处理效率提升技巧

Fun-ASR-MLT-Nano-2512性能优化&#xff1a;批量处理效率提升技巧 1. 引言 1.1 业务场景与技术背景 在多语言语音识别的实际应用中&#xff0c;Fun-ASR-MLT-Nano-2512 凭借其对31种语言的高精度支持和轻量化设计&#xff0c;成为边缘设备和中小规模服务部署的理想选择。该模…

作者头像 李华