C++ 泛编程 —— 函数模板(中)

C++ 泛编程 ------ 函数模板(中)

函数模板的具体化

可以提供一个具体化的函数定义,当编译器找到与函数调用匹配的具体化定义时,将使用该定义,不再寻找模板。
具体化(特例化、特化)的语法:

cpp 复制代码
template<> void 函数模板名<数据类型>(参数列表)
template<> void 函数模板名 (参数列表)
{
	// 函数体。
}

请看如下示例代码:

cpp 复制代码
#include <iostream>
using namespace std;

class Car{
public:    
    string brand_;
    int speed_;
    float acceleration_;
};

template<typename T> 
void change(T& t1, T& t2){
    T tmp = t1;
    t1 = t2;
    t2 = tmp;
}

// 现在的需求是:只交换 Car 类对象中的 speed_ 属性值,不交换其他属性值。
// 具体化函数的代码如下
// template<> void change<Car>(Car& c1, Car& c2) // 第一种写法
template<> void change(Car& c1, Car& c2) { // 第二种写法 交换两个Car类对象的speed_属性值
    int tmp = c1.speed_;
    c1.speed_ = c2.speed_;
    c2.speed_ = tmp;
}

int main(){
    int a = 10, b = 20;
    change<int>(a, b);
    cout << "a = " << a << ", b = " << b << endl;
    Car c1 = {"BMW", 20, 3.5};
    Car c2 = {"Audi", 30, 4.5};
    change(c1, c2);
    cout << "c1.brand_ = " << c1.brand_ << ", c2.brand_ = " << c2.brand_ << endl;
    cout << "c1.speed_ = " << c1.speed_ << ", c2.speed_ = " << c2.speed_ << endl;

    return 0;
}

编译运行的结果如下:

a = 20, b = 10

c1.brand_ = BMW, c2.brand_ = Audi

c1.speed_ = 30, c2.speed_ = 20

对于具体化函数来说,不管是函数的声明还是函数体中,都是具体的数据类型,没有通用的数据类型了。

对于给定的函数名,它可以是普通函数函数模板具体化的函数模板,以及它们的重载版本。如果多种函数都能匹配上,那编译器会使用哪个函数呢?编译器使用各种函数的规则:

  1. 具体化优先于常规模板,普通函数优先于具体化和常规模板。
  2. 如果希望使用函数模板,可以用空模板参数强制使用函数模板。
  3. 如果函数模板能产生更好的匹配,将优先于普通函数。

对于前两条规则,给出示例代码,大家可以自行选择注释某个版本的函数代码,来测试一下各个版本函数代码的使用规则:

cpp 复制代码
#include <iostream>
using namespace std;

// 普通版本函数
void show(int n1, int n2) {
    cout << "使用普通版本函数" << endl;
}

// 常规函数模板
template <typename T>
void show(T n1, T n2) {
    cout << "使用常规函数模板" << endl;
}

// 函数模板的具体化版本
template <>
void show(int n1, int n2) {
    cout << "使用函数模板的具体化版本" << endl;
}

int main() {
    show(111, 222);
    show<>(111, 222);

    return 0;
}

编译运行结果如下:

使用普通版本函数

使用函数模板的具体化版本

对于第三条规则:如果函数模板能产生更好的匹配,将优先于普通函数,示例代码如下:

cpp 复制代码
#include <iostream>
using namespace std;

// 普通版本函数
void show(int n1, int n2) {
    cout << "使用普通版本函数" << endl;
}

// 常规函数模板
// template <typename T>
// void show(T n1, T n2) {
//     cout << "使用常规函数模板" << endl;
// }

int main() {
    show('a', 'b'); // 字符可以转换成整数

    return 0;
}

编译运行结果如下:

使用普通版本函数

如果取消注释,编译运行结果如下:

使用常规函数模板

原因解释:实参是字符ab,与常规函数模板可以完全匹配,不需要转换,比普通函数匹配度更好。所以,编译器选择了常规函数模板。

C++ 泛编程 ------ 函数模板(上)

C++ 泛编程 ------ 函数模板(下)

感谢浏览,一起学习!

相关推荐
梓䈑3 分钟前
【Linux系统】实现线程池项目(含日志类的设计)
linux·服务器·c++
Ashley_Amanda14 分钟前
Python 进阶:从熟练到精通的核心技能体系
开发语言·python
你怎么知道我是队长20 分钟前
C语言---命令行参数
c语言·开发语言
秋刀鱼程序编程20 分钟前
Java编程基础入门(四)---选择循环语句
java·开发语言·算法
优雅的潮叭22 分钟前
c++ 学习笔记之 volatile与atomic
c++·笔记·学习
一条咸鱼_SaltyFish22 分钟前
WebFlux vs MVC:Gateway集成若依框架的技术选型之争
java·开发语言·微服务·gateway·mvc·开源软件·webflux
独自归家的兔31 分钟前
Java反射之根:Class类生成机制深度剖析与最佳实践
java·开发语言
请叫我聪明鸭1 小时前
基于 marked.js 的扩展机制,创建一个自定义的块级容器扩展,让内容渲染为<div>标签而非默认的<p>标签
开发语言·前端·javascript·vue.js·ecmascript·marked·marked.js插件
仍然.1 小时前
Java---反射、枚举、lambda表达式 和 泛型进阶
java·开发语言
Zsy_0510031 小时前
【C++】类和对象(二)
开发语言·c++