首页 > 编程语言 >c++ 模板编程

c++ 模板编程

时间:2024-12-09 19:54:27浏览次数:6  
标签:函数 int 代码 编程 c++ template 模板 特化

c++ 模板编程

C++中模板分为函数模板和类模板

函数模板:是一种抽象函数定义,它代表一类同构函数。

类模板:是一种更高层次的抽象的类定义。

优缺点

优点

  1. 代码复用
      模板允许编写与具体数据类型无关的代码,从而实现代码复用。你可以针对不同的数据类型使用相同的模板函数或模板类,而无需为每种数据类型重复编写代码。
  2. 类型安全
      模板是在编译时生成代码的,编译器会对模板实例化的代码进行严格的类型检查,从而确保类型安全,避免运行时类型错误。
  3. 性能优化
      模板在编译时会生成具体类型的代码,相当于为每种使用的类型生成了定制的函数或类。这种机制避免了运行时多态的开销,性能接近手动为每种类型编写的代码。
  4. 支持泛型编程
      模板是 C++ 泛型编程的核心工具,使得开发者能够以声明性和抽象的方式定义算法和数据结构。例如,标准模板库(STL)中的容器和算法就是基于模板实现的。
  5. 灵活性强
      模板支持元编程(template metaprogramming),允许在编译时执行复杂的计算和逻辑。这为高级编程提供了强大的能力,比如静态多态和编译时优化。
  6. 消除冗余代码
      使用模板可以大大减少重复代码,降低维护成本。例如,可以为不同类型的数据结构(如 std::vector 和 std::vector)使用相同的模板定义,而无需单独实现。

缺点

  1. 编译时间长
      模板会在每次实例化时生成具体的代码,这可能导致编译时间显著增加,尤其是在大型项目中使用了大量模板时。
  2. 可读性和可维护性差
      模板代码往往较为复杂,尤其是涉及到嵌套模板、模板元编程或 SFINAE(Substitution Failure Is Not An Error)时,代码的可读性和可调试性会大大降低。
  3. 代码膨胀(Code Bloat)
      模板会为每种使用的类型生成一份实例化代码,可能导致最终的二进制文件体积显著增加。虽然现代编译器可以通过模板共享(template pooling)来优化,但这个问题仍然存在。
  4. 错误信息复杂
      模板相关的编译错误通常非常复杂且难以理解,尤其是当涉及到深度嵌套的模板或复杂的模板元编程时。这会增加调试和排错的难度。
  5. 二进制接口(ABI)兼容性问题
      模板代码是基于实例化生成的,不同编译器或编译器版本可能会生成不同的二进制代码,导致模板类或函数难以在动态链接库(DLL/so)之间共享。
  6. 限制动态多态
      模板实例化的代码在编译时就确定了具体的类型,因此不支持运行时的动态类型分派(如虚函数机制)。这使得模板更适合静态多态,而不适合动态多态。
  7. 缺乏显式约束(在 C++20 前)
      在 C++20 之前,模板参数的约束依赖隐式的接口契约,编译器只在实例化时检查是否满足接口要求,这可能导致错误信息迟滞且难以定位。
      C++20 引入了 Concepts,显著改善了这一问题,但在旧版本中,约束模板的机制仍显不足。
  8. 调试困难
      模板代码在调试工具中可能表现为一大堆展开后的代码片段,使得调试过程变得复杂且耗时。
  9. 潜在的库升级问题
      如果一个库的大量接口通过模板实现,那么升级库可能导致用户代码的重新编译,因为模板实例化通常是内联的。

模版特化

需要对某些参数的模版进行特殊处理,此时可以用特化

全特化

  通过全特化一个模板,可以对一个特定参数集合自定义当前模板,类模板和函数模板都可以全特化。 全特化的模板参数列表应当是空的,并且应当给出"模板实参"列表:

// 全特化类模板
template <>
class A<int, double>{
    int data1;
    double data2;
};

// 函数模板
template <>
int max(const int lhs, const int rhs){   
    return lhs > rhs ? lhs : rhs;
}

  注意类模板的全特化时在类名后给出了"模板实参",但函数模板的函数名后没有给出"模板实参"。 这是因为编译器根据int max(const int, const int)的函数签名可以推导出来它是T max(const T, const T)的特化。

特化的歧义

  上述函数模板不需指定"模板实参"是因为编译器可以通过函数签名来推导,但有时这一过程是有歧义的:

template <class T>
void f(){ T d; }

template <>
void f(){ int d; }

此时编译器不知道f()是从f()特化来的,编译时会有错误:

error: no function template matches function template specialization 'f'

这时我们便需要显式指定"模板实参":

template <class T>
void f(){ T d; }

template <>
void f<int>(){ int d; }

偏特化

类似于全特化,偏特化也是为了给自定义一个参数集合的模板,但偏特化后的模板需要进一步的实例化才能形成确定的签名。 值得注意的是函数模板不允许偏特化

template <class T2>
class A<int, T2>{
    ...
};

函数模板是不允许偏特化的,下面的声明会编译错:

template <class T1, class T2>
void f(){}

template <class T2>
void f<int, T2>(){}

但函数允许重载,声明另一个函数模板即可替代偏特化的需要:

template <class T2>
void f(){}              // 注意:这里没有"模板实参"

多数情况下函数模板重载就可以完成函数偏特化的需要,一个例外便是std命名空间。 std是一个特殊的命名空间,用户可以特化其中的模板,但不允许添加模板(其实任何内容都是禁止添加的)。 因此在std中添加重载函数是不允许的,在Effective C++: Item 25中给出了一个更详细的案例。

使用总结

  • 模板类的成员函数要和类放在同一个文件内

继承模板类调用父类的函数要加 this 或者 base:: 显示指定,或者用 using base::xxfunc; bast:: 如果是虚函数会破会多态性

模板函数要声明为inline,可以节省内存

事实上类型T::const_iterator依赖于模板参数T, 模板中依赖于模板参数的名称称为从属名称(dependent name), 当一个从属名称嵌套在一个类里面时,称为嵌套从属名称(nested dependent name)。 其实T::const_iterator还是一个嵌套从属类型名称(nested dependent type name)。

嵌套从属名称是需要用typename声明的,其他的名称是不可以用typename声明的。

标签:函数,int,代码,编程,c++,template,模板,特化
From: https://www.cnblogs.com/AngleLin/p/18595904

相关文章

  • C++小小复习一下
    类,对象,成员变量,成员函数特点:面向对象程序设计---因为要创建对象来调用类里面的函数或者成员变量比如你的对象是一个生物-人:他会有自己的一些属性:身高,体重,性别等,还有一些行为比如:有人惹他,他会骂回去或者打回去,这个骂和打的动作或者行为需要一系列的身体各个机能共同合作才能实......
  • C++_默认构造函数和重载以及设计模式
    类和类之间关系类外:静态变量,是在编译阶段就分配好空间,对象还没创建的时候就有了空间类:类-对象-对象是类的一个实例类头(classhead)和类体(classbody)。将数据和行为封装在单个单元中--封装成员变量成员函数成员变量称为属性(property);成员函数称为方法(metho......
  • [c++]c++ 工程代码中的debug时条件编译隐去的代码会影响程序运行的性能和耗时吗
    前言 理解在C++(或任何编程语言)中,使用条件编译(如通过#ifdef,#ifndef,#endif预处理指令)来根据调试(debug)或发布(release)模式包含或排除代码段,对程序在最终编译后的性能和耗时通常是没有直接影响的。这是因为条件编译指令是在编译之前处理的,它们决定了哪些代码会被编译器实际编......
  • 第七章:C#响应式编程System.Reactive
    第七章:C#响应式编程System.Reactive目录第七章:C#响应式编程System.Reactive7.1为什么选择响应式编程?1.事件流的重要性2.Rx.NET的优势3.Rx.NET的适用场景4.Rx.NET的核心思想小结7.2主要概念和类型1.IObservable<T>2.IObserver<T>3.PushvsPull(推vs拉)4.热(Hot)和冷(Co......
  • 【人工智能】Moss-AI编程利器:CodeMoss & ChatGPT中文版超详细入门教程!(VScode/IDER/WE
    文章目录摘要一、环境介绍VSvode安装步骤IDER(Pycharm)安装步骤Web使用步骤二、Moss9大功能讲解1、AI问答对话2、文件上传功能3、自定义AI助手4、AI联网助手5、AI图片识别6、思维链思维链的简单介绍使用CodeMoss思维链7、AI图片生成图片生成效果8、图片生成代码9、......
  • Java模板引擎 Velocity
    目录前言1.Velocity模板引擎概述1.1什么是Velocity1.2Velocity的特点1.3Velocity的主要应用场景2.Velocity的核心原理2.1模板准备2.2数据填充2.3渲染输出3.Velocity的基本使用方法3.1环境配置3.2编写模板3.3数据填充与渲染3.4常见功能示例4.Velocity在......
  • 第五章:C#并行编程
    第五章:C#并行编程基础目录第五章:C#并行编程基础5.1并行处理:使用Parallel.ForEach和Parallel.For问题解决方案示例1:并行计算大量数字的平方根示例2:提前终止并行计算示例3:并行计算时使用CancellationToken示例4:处理共享状态Parallel.For示例:并行处理数组小结5.2并行聚合......
  • 题解:P11266 【模板】完全体·堆
    题目链接洛谷P11266【模板】完全体·堆解题思路看了题解区,竟然没有人写可爱的左偏树。我们需要支持四种操作:删除集合中的元素。取集合中的最小值。合并两个集合。修改集合中的元素。那么我们可以用常数极小的左偏树(可并堆)来解决这道模板题。对于左偏树的基础操作......
  • 【Java编程】深入解析 Java 的四大引用类型:强引用、软引用、弱引用、虚引用
    在Java内存管理中,引用的概念扮演着非常重要的角色。引用的强弱程度,直接影响对象在内存中的生存周期。Java语言中定义了四种不同的引用类型:强引用、软引用、弱引用和虚引用,每一种引用类型都有独特的特性和适用场景。理解它们的区别和作用,不仅有助于提升Java内存管理的水平,也有......
  • 【SpringBoot 编程】在SpringBoot中拦截修改请求Body的2种正确方式
    环境:SpringBoot3.2.51.简介修改请求Body内容的需求源于多种场景,其中最重要的是数据预处理和安全性考虑。在Web应用中,客户端发送的请求数据可能不符合后端服务的直接处理要求,如格式不匹配、不文明用语、数据不完整或包含敏感信息。通过修改请求Body,可以在数据到达Controller之......