详解 C++中的模板

server/2024/9/25 4:39:06/

目录

前言

一、函数模板

1.定义

2.函数模板的实现

3.模板函数的实例化

4.模板参数的省略

1.函数模板的实参推导

2.类模板的实参推导

3.默认模板参数

4.特殊情况:无法推导的模板

5.推导失败的情况

二、类模板

1.概念和定义

2.类模板定义

3.类模板的使用

4.类模板的定义格式


前言

        这篇文章主要介绍C++中的模板。

一、函数模板

1.定义

        模板是一系列相关函数的模型或样板,这些函数的源代码形式相同,只是所针对的数据类型不同。

        在开发过程中,我们经常遇到这种情况,有两个或者两个以上的函数,其功能是相同的,仅仅是数据类型的不同。例如:

int add(int a,int b){return a + b;
}
int add(float a,float b){return a + b;
}double add(double a,double b){return a + b;
}

        上面的三个函数都实现了对数据的加法运算,唯一不同的是参数类型不同。这样的函数可以采用函数模板实现简便化。

2.函数模板的实现

        函数模板的格式如下:

template <<模板形参声明>> <函数声明>

        模板形参是由一个或者多个<模板形参>组成的,如果是多个需要用逗号隔开。每个模板具有以下几种形式:

1.typename <参数名>

2.class <参数明>

3.<类型修饰><参数名>

        对于上述实例中的 add 函数,我们可以如下定义:

template <typename T>
T add(T a, T b) {return a + b;
}

3.模板函数的实例化

        在实际开发过程中,我们使用实参的实际类型代替虚拟类型即可。

#include <iostream>
#include <iomanip> // 需要包含这个头文件来设置精度
using namespace std;template <typename T>
T add(T a, T b) {return a + b;
}int main() {int a = 10, b = 20;cout << "a + b = " << add(a, b) << endl;// 使用浮点数并设置精度double c = 10.0, d = 20.0;cout << fixed << setprecision(2);  // 设置保留两位小数cout << "c + d = " << add(c, d) << endl;double e = 10.00, f = 20.00;cout << "e + f = " << add(e, f) << endl;return 0;
}

4.模板参数的省略

        在C++中,模板实参可以在某些情况下进行省略,称为 模板实参推导。这一特性允许编译器根据传递给函数或类的参数自动推导出模板的类型。以下是几种常见的模板实参省略场景:        

1.函数模板的实参推导

        对于函数模板,编译器可以根据调用时传递的函数参数来推导模板实参。例如:

template <typename T>
T add(T a, T b) {return a + b;
}int main() {int x = 5, y = 10;// 不需要显式指定类型,编译器会推导T为intcout << add(x, y) << endl;
}

        在上述例子中,add(x, y) 调用时,编译器会根据 x 和 y 的类型(int)自动推导 T 的类型为 int。

2.类模板的实参推导

        在C++17之前,类模板的实参必须显式指定,但从C++17开始,可以省略某些类模板的实参。编译器会从构造函数参数中推导出模板实参。例如:

template <typename T>
T add(T a, T b) {return a + b;
}int main() {int x = 5, y = 10;// 不需要显式指定类型,编译器会推导T为intcout << add(x, y) << endl;
}

        在这个例子中,Box box(123); 中并没有显式地指定 Box<int>,编译器根据传递的值 123 推导出 T 的类型为 int。

3.默认模板参数

        你可以为模板提供默认的模板参数,这样在不提供实参时会使用默认值。例如:

template <typename T = int>
T multiply(T a, T b) {return a * b;
}int main() {cout << multiply(3, 4) << endl; // T被推导为int,因为int是默认类型cout << multiply<double>(3.5, 2.5) << endl; // T显式为double
}

        在这个例子中,multiply(3, 4) 直接使用了默认的模板参数 T = int,而 multiply<double> 明确指定了模板参数。

4.特殊情况:无法推导的模板

        并非所有情况下模板都可以自动推导,特别是当模板类型不直接与参数关联时。例如,某些模板的类型依赖于非参数部分时,编译器无法自动推导,这时需要显式指定模板实参。

template <typename T, typename U>
void printPair(T a, U b) {cout << a << " and " << b << endl;
}int main() {printPair(1, "hello"); // 编译器能够推导T为int,U为const char*printPair<int>("test", 100); // 必须显式指定其中一个模板参数
}

5.推导失败的情况

        有时候模板的推导会失败,特别是类型不匹配或涉及复杂的类型转换时。此时需要显式指定模板实参来避免推导失败。

二、类模板

1.概念和定义

        类模板是通过引入模板参数来定义类的。在类模板的声明中,使用 template 关键字标识模板类型参数,然后在类的定义中可以使用这些模板参数,就像使用普通的数据类型一样。

2.类模板定义

        一个简单的类模板可以按如下方式定义:

template <typename T>
class Box {
private:T value;
public:Box(T val) : value(val) {}T getValue() {return value;}
};

        在上述代码中,T 是一个模板参数,可以被替换为任何具体的数据类型(例如 int、double 或 string)。当我们使用这个类时,必须在实例化类时提供一个具体的数据类型。

3.类模板的使用

        在这个例子中,Box<int> 和 Box<double> 分别创建了 int 和 double 类型的对象,并且它们可以分别存储 int 和 double 类型的数据。        ​​​​​​​        

int main() {Box<int> intBox(100);  // 使用int类型Box<double> doubleBox(100.5);  // 使用double类型std::cout << "intBox value: " << intBox.getValue() << std::endl;std::cout << "doubleBox value: " << doubleBox.getValue() << std::endl;return 0;
}

4.类模板的定义格式

        类模板的定义格式如下:

template <typename T> 
class ClassName {// 成员变量和方法使用模板参数T
};

        类模板的优势在于能够通过一种通用方式处理不同类型的数据,使代码更具通用性和灵活性,适用于需要同样逻辑但适用于不同数据类型的情况。


http://www.ppmy.cn/server/121661.html

相关文章

23个Python在自然语言处理中的应用实例

在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;Python作为一门功能强大的编程语言&#xff0c;凭借其丰富的库和工具集&#xff0c;成为了实现各种NLP任务的首选。以下是一个关于Python在NLP中应用的广泛实例的前言&#xff0c;旨在概述Python在NLP领域的多样性和…

力扣 中等 1901.寻找峰值II

文章目录 题目介绍题解 题目介绍 题解 需要明白一个事实&#xff1a;从任意一个点出发&#xff0c;可以经过一个递增路径&#xff0c;找到一个极大值点。 求出一行的最大值&#xff0c;如果这行最大值比上面的要小&#xff0c;那峰值&#xff08;之一&#xff09;就会在上面 …

sqoop的安装与简单使用

文章目录 一、安装1、上传&#xff0c;解压&#xff0c;重命名2、修改环境变量3、修改配置文件4、上传驱动包5、拷贝jar包 二、import命令1、将mysql的数据导入到hdfs上2、将mysql的数据导入到hive上3、增量导入数据 三、export命令1、从hdfs导出到mysql中2、从hive导出到mysql…

Linux之实战命令02:shred应用实例(三十六)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列【…

canvas练习画太阳花

// file name: demo.html <!DOCTYPE html> <html><head><meta charset"UTF-8"><title>练习</title><style>#myCanvas {box-shadow: 6px 9px 80px 13px rgba(0,0,0,.65);margin: 50px 400px;}</style></head>&…

Python数据分析案例60——扩展变量后的神经网络风速预测(tsfresh)

案例背景 时间序列的预测一直是经久不衰的实际应用和学术研究的对象&#xff0c;但是绝大多数的时间序列可能就没有太多的其他的变量&#xff0c;例如一个股票的股价&#xff0c;还有一个企业的用电量&#xff0c;人的血糖浓度等等&#xff0c;空气的质量&#xff0c;温度这些…

AI Native平台,跨越AI应用从创新到生产的鸿沟

2024年是AI应用的元年&#xff0c;以大模型为中心的 AI Native 应用大爆发正在从理想变成现实。云计算带来的应用创新潮&#xff0c;经历了虚拟机时代和云原生时代&#xff0c;正在全面拥抱以大模型为核心的 AI Native 阶段&#xff0c;推动大数据与AI的工作流前所未有地紧密结…

Cryptopp进行RSA加密解密

因最近工作需要&#xff0c;要与一个java服务器进行部分数据交互&#xff0c;其中用户名、密码的加密方式为RSA非对称加密&#xff0c;故研究了一下使用Cryptopp库进行RSA的公钥加密。 搜索了很多资料&#xff0c;其中觉得十分有用的一句话是&#xff1a; 客户端使用RSA加密通…