C++模板之模板成员函数不能偏特化

news/2024/7/5 7:51:15

目录

1.引言

2.类模板成员函数的特化

2.1.没有函数特化的类模板

2.2.增加函数特化        

3.“曲线救国”函数“偏特化”

3.1.函数重载实现“偏特化”

3.2.使用类型选择机制实现“偏特化”

4.总结


1.引言

        C++ 泛型编程的资料在介绍类模板的特化和偏特化的时候,都会提到“虽然类模板可以偏特化,但是类模板的成员函数不可以偏特化”。对于初次接触泛型编程的 C++ 程序员来说,甚至对 C++ 有一定了解的人,一时也会摸不着头脑。到底模板成员函数的特化和偏特化是怎么回事儿,类模板成员函数不能偏特化是什么意思?本文就用几个简单的例子解释一下这句话。

2.类模板成员函数的特化

2.1.没有函数特化的类模板

        当你开始了解 C++ 的泛型编程的时候,类模板的特化和偏特化就是两个绕不开的话题。在很多 C++ 的书籍中都会提到一点,就是:“虽然类模板可以偏特化,但是类模板的成员函数不可以偏特化”。这句话到底是什么意思呢?本篇小文就用一段例子代码,直观地解释一下这句话。读懂本篇小文,你需要对 C++ 的类模板、类模板的特化相关的概念和 C++ 的语法实现有基本的了解。
        废话少说,我们先定义一个类模板 SpecialClass<> ,它有两个模板参数,分别是 U 和 T。这个类模板有两个(函数)方法,即 FuncA() 和 FuncB(),如下代码片段所示:

        对于这个类模板,我们可以这样实例化它们,下面的代码片段中,sp1 和 sp2 就是类模板的两个实例化对象:

        不出意料,两个类模板的实例对象的函数调用的输出结果都是主模板(Primary Template)类的 FuncA() 和 FuncB() 的输出:

SpecialClass::FuncA(U, T) invoked!
SpecialClass::FuncB(U, T) invoked!
SpecialClass::FuncA(U, T) invoked!
SpecialClass::FuncB(U, T) invoked!

2.2.增加函数特化        

        既然标准已经说了,类模板不支持成员函数的偏特化,那么我们先来个全特化成员函数,看看全特化成员函数是什么情况。我们对 FuncA() 进行全特化,保留 FuncB() 作为对比:

        我们用 int 和 char 对 FuncA() 进行特化,然后再像代码 2 那样创建两个类模板的实例,此时运行的结果如下所示,第二个实例中的 FuncA() 特化起了作用,但是 FuncB()仍然是主模板类的函数:

SpecialClass::FuncA(U, T) invoked!
SpecialClass::FuncB(U, T) invoked!
SpecialClass::FuncA(int, char) invoked!
SpecialClass::FuncB(U, T) invoked!

        正如标准所说的那样,类模板中函数的全特化是没有问题的,那么我们现在来看看函数的偏特化。如下代码片段所示,我们只特化模板参数 T:

        终于,编译器拒绝编译这段代码,不同的编译器可能给出的错误信息不一样。我用的 gcc 7.3 抱怨说 FuncA 的类型 class SpecialClass<U, char> 声明不完整,Visual C++ 14.0(Visual Studio 2015) 给出的错误信息是SpecialClass<U,char>::FuncA 无法匹配已经存在的函数声明:void SpecialClass<U,T>::FuncA(U,T) 。编译器的提示信息和大多数涉及模板的编译错误提示信息一样,一如既往的含蓄和隐晦。
        至此,我们已经用具体的例子解释了类模板成员函数的特化和偏特化(不被允许),实际上,不仅类模板的成员函数不支持偏特化,有具体的名字域(namespace)级别的非成员函数,也不支持偏特化。如果我们不死心,一定要 FuncA 的第二个参数使用 char 类型怎么办呢?方法当然有,那就是利用 C++ 语言的函数重载机制“曲线救国”。

3.“曲线救国”函数“偏特化”

        注意,这里要介绍的两个方法都不是真正意义上的函数偏特化,因为 C++ 不支持模板函数偏特化,所以标题都加了引号。这里介绍的两种方法都是采用模拟的方法,让编译器能够根据函数参数的差异实现函数的最佳匹配。

3.1.函数重载实现“偏特化”

        假如我们无论如何都要给 SpecialClass<> 增加一个 FuncA 的定制版本: FuncA(U para1, char para2),使得这个函数对 char 类型参数时,能够使用更高效的算法实现。简单一点的方法就是利用函数重载,直接给 SpecialClass<> 增加一个重载函数,比如这样:

template <class U, class T>
class TestClass
{
public:
    void FuncA(U para1, T para2)
    {
        std::cout << "TestClass::FuncA(U, T) invoked!" << std::endl;
    }
    void FuncA(U para1, char para2)
    {
        std::cout << "TestClass::FuncA(U, char) invoked!" << std::endl;
    }
};

//模板实例化应用
TestClass<std::string, double> tp1;

tp1.FuncA("dsdfsd", 23.1); //第一个重载函数被调用
tp1.FuncA("dsdfsd", 'B'); //第二个重载函数被调用

3.2.使用类型选择机制实现“偏特化”

C++ 的 Tag Dispatching(标签派发) 惯用法-CSDN博客

        《Modern C++ Design》这本书里介绍了一种利用编译期间类型选择机制实现的“函数”的偏特化选择,这种技术被称为“Tag Dispatching”,其本质仍然是利用了 C++ 的函数重载机制,这里我们就简单介绍一下这种技术。
        首先介绍一种 C++ 泛型编程种常用的技巧,即“类型对类型的映射”。C++ 是一种强类型语言,对于类模板的实例来说,SomeClass<int> 和 SomeClass<double> 是两种不同的类型(虽然它们的代码是一套代码)。利用这个特点,我们可以定义一个类型到类型映射的模板类:

template <typename T>
struct Type2Type
{
    typedef T thisNewType;
};

        这里我想插入解答一个读者经常问我的问题,即:为什么你这里用 struct 而不是 class?其实不只是我,很多人都这么用,原因很简单,就是省事儿,你接着往下看就明白了。这样在获取映射类型的时候,就可以直接这么用:

Type2Type<T>::thisNewType 是个类型定义,可被用于编译器的自动推导

Type2Type<int>::thisNewType a; // 等同于: int a;

        当然可以用 class ,不过别忘了,C++ 的 class 与 struct 的区别之一就是 class 的属性默认是私有的,如果用 class,代码要这么写:

template <typename T>
class Type2Type
{
public:
    typedef T thisNewType;
};

        好了,现在言归正传。我们已经知道 Type2Type<int> 和 Type2Type<double> 是两种完全不同的类型,如果将其作为函数的参数,就可以利用函数的重载机制,让编译器区分它们,既能够根据类型自动生成对应类型的函数实例,又能引导编译器将某些特定类型参数的函数调用链接到针对特定类型实现的函数体上。

template <class U, class T>
T* CreateObject(U para1, Type2Type<T>)
{
    std::cout << "SpecialTypeClass::CreateObject(U, T) invoked!" << std::endl;
    return new T(para1); //毫无意义的代码
}

template <class U>
char* CreateObject(U para1, Type2Type<char>)
{
    std::cout << "SpecialTypeClass::CreateObject(U, char) invoked!" << std::endl;
    return new char(para1); //又一行毫无意义的代码
}

       使用起来也很简单,第二个参数仅仅是用于告诉编译器如何根据类型匹配正确的函数体:

double *pValue = CreateObject(5, Type2Type<double>());

char *pCh = CreateObject(5, Type2Type<char>());

       没有悬念,程序输出如我们所愿:

SpecialTypeClass::CreateObject(U, T) invoked!
SpecialTypeClass::CreateObject(U, char) invoked!

        使用这种方法,还有一个好处,就是简化 CreateObject() 函数的使用。想想在 C++11 标准出来之前,为了让编译器能正确实作出对应模板参数的函数,我们不得不这样写代码:

//这样定义模板函数
template <class U, class T>
T* CreateObject(U para1)
{ 
    std::cout << "SpecialTypeClass::CreateObject(U, T) invoked!" << std::endl; 
    return new T(para1); //毫无意义的代码
}

//这样使用模板函数
double *pValue = CreateObject<int, double>(5);

4.总结

        好了,本篇小文就写到这里,总结一下内容,首先实例代码讲解了很多 C++ 书籍都提到,但是没有细说的:“虽然类模板可以偏特化,但是类模板的成员函数不可以偏特化” 这句话的意思,然后又介绍了两种利用函数重载机制实现的变相函数“偏特化”的方法。


http://lihuaxi.xjx100.cn/news/2215821.html

相关文章

【在线OJ】vue分页+SpringBoot分页模板代码

一、Vue <template><div><el-table:data"user"style"width: 120%"><el-table-columnlabel"id"width"180"><template slot-scope"scope"><i class"el-icon-time"></i>&…

C# String

String字符串字面量字符串连接字符串是不可变的字符串比较字符串方法字符串插值字符串和字符数组字符串格式化空字符串和 null字符串的安全性正则表达式 注意String.Format基本语法参数 基本使用使用索引指定对齐和宽度使用格式字符串组合使用 总结 C# String 在C#中&#xff0…

Elasticsearch 8.1官网文档梳理 - 十一、Ingest pipelines(管道)

Ingest pipelines 管道&#xff08;Ingest pipelines&#xff09;可让让数据在写入前进行常见的转换。例如可以利用管道删除文档&#xff08;doc&#xff09;的字段、或从文本中提取数据、丰富文档&#xff08;doc&#xff09;的字段等其他操作。 管道&#xff08;Ingest pip…

牛客小白月赛96 D 最小连通代价

题目在这里 题意: 加边是所有点连通&#xff0c;没有重边和自环,问最小代价 加边规则&#xff1a;两点权值奇偶性相同代价为a,否则为b − 100 ≤ a , b ≤ 100 -100\leq a,b \leq100 −100≤a,b≤100 分析&#xff1a; 这题就是一个分类讨论,先读进来统计奇数点和偶数点 记 …

Temu跨境电商按关键字搜索Temu商品API的应用及接口请求示例

Temu跨境电商按关键字搜索Temu商品API的应用场景 Temu跨境电商平台按关键字搜索Temu商品API的主要应用场景包括但不限于以下几个方面&#xff1a; 用户搜索商品&#xff1a;当用户在Temu平台上输入关键字搜索商品时&#xff0c;API会根据输入的关键字返回与之相关的商品列表&a…

Spring框架的原理及应用详解(五)

本系列文章简介&#xff1a; 在当今的软件开发世界中&#xff0c;随着应用复杂性的不断增加和技术的快速发展&#xff0c;传统的编程方式已经难以满足快速迭代、高可扩展性和易于维护的需求。为此&#xff0c;开发者们一直在寻求更加高效、灵活且易于管理的开发框架&#xff0c…

【杂记-浅谈以太网IP数据帧】

一、以太网数据帧 以太网数据帧是网络通信的基础单元&#xff0c;遵循IEEE 802.3标准&#xff0c;用于在以太网中传输数据。一个典型的以太网数据帧包括前导码、帧开始符、目的MAC地址、源MAC地址、类型或长度字段、数据载荷和帧校验序列。其中&#xff0c;类型字段指明了上层…

Scala网络编程:代理设置与Curl库应用实例

在网络编程的世界里&#xff0c;Scala以其强大的并发模型和函数式编程特性&#xff0c;成为了开发者的得力助手。然而&#xff0c;网络请求往往需要通过代理服务器进行&#xff0c;以满足企业安全策略或访问控制的需求。本文将深入探讨如何在Scala中使用Curl库进行网络编程&…