C++ 的 Tag Dispatching(标签派发) 惯用法

2024-06-02 17:28

本文主要是介绍C++ 的 Tag Dispatching(标签派发) 惯用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1.概述

2.标准库中的例子

3.使用自己的 Tag Dispatching

3.1.使用 type traits 技术

3.2.使用 Type_2_Type 技术

4.Tag Dispatching的使用场景

5.总结


1.概述

        一般重载函数的设计是根据不同的参数决定具体做什么事情,编译器会根据参数匹配的原则确定正确的重载版本。但是对于函数模板,其参数类型是泛化的模板参数,此时又如何让编译器选择我们希望的那个函数模板的实例呢?提供特化版本是一个方法,但是如果需要特殊处理的类型很多,就需要搞一大堆特化版本,非常不方便。C++ 11 的语言库提供了 std::enable_if,配合编译器的 SFINAE 原则也可以实现在编译期间的特定选择。C++ 17 还提供了一个 std::void_t,以模板别名定义的语法形式提供了另一种利用 SFINAE 的方法。当然,同样是 C++ 17 提供的 if constexpr 语言特性配合各种 type traits,可以更优雅地实现编译期间的特定选择。但是这一篇我们要介绍的是另一种常用的习惯用法(或技术):Tag Dispatching

        在C++中,标签分发(Tag Dispatching)或标签分派是一种技术,它允许你根据传递给函数的参数类型或某个特定标签来选择不同的函数或函数模板进行执行。这通常用于实现重载函数的泛型版本,其中你可能需要根据参数的某些特性(如类型、状态等)来执行不同的逻辑。

        Tag Dispatching 是一种利用某种类型特征,在一系列重载函数之间进行编译期调度(分派、选择)的技术。Tag Dispatching 并不是 C++ 的某种特性,但是作为一种习惯用法在 C++ 中被广泛应用,尤其是在标准库中。这里说的 tag,其实就是定义一种没有操作、没有数据的类型,将这种类型作为重载函数的一个参数,通过不同的 tag 参数控制编译器的选择。定义一个 tag 非常简单,一般用 struct:

struct tag1 {};
struct tag2 {};

        虽然结构体都是空的,但是在 C++ 编译器看来,tag1 和 tag2 是两个完全不同的类型。基于 Tag Dispatching 的实现就是定义不同的 tag,并将 tag 设计成函数的一个参数。一般会将 tag 设计成 函数的最后一个参数,因为编译器在代码生成的时候对这种完全是空的参数类型会有针对性的优化。具体来说,就是将重载函数设计成这个样子:

template <typename T>
int Function(T t, tag1) { ... }template <typename T>
int Function(T t, tag2) { ... }

这就是所谓的 Tag Dispatching,其实就是利用 tag1 和 tag2 是不同类型的特性,控制编译器在编译期间选择希望的重载版本,实现在编译期间的重载分派,比如:

int a = Function(42, tag1());

可以确保编译器使用第一个模板函数。这只是一个简单的例子,要让编译器能够根据类型自动选择,还需要自定义 type traits,请继续看下去。

2.标准库中的例子

        标准库中大量使用 Tag Dispatching,这一节就介绍一下标准库的 std::advance() 函数。void std::advance(Iter& it, Distance n) 函数的作用是将迭代器向前(或向后)移动 n 个位置。这里需要注意的是,根据迭代器类型的不同,std::advance() 函数内部是不同的实现。比如对于随机类型的迭代器,可以采用高效的 it + n 的形式移动位置,对于不支持随机访问的单向迭代器,只能通过执行 n 次 ++it 的方式移动迭代器,而对于双向类型的迭代器,n 可以是负数,表示向后移动迭代器。

        std::advance() 函数首先针对不同类型的迭代器定义了相应的重载形式:

template <class RAIter, class Distance>
void advance(RAIter& it, Distance n, std::random_access_iterator_tag) {it += n;
}template <class BidirIter, class Distance>
void advance(BidirIter& it, Distance n, std::bidirectional_iterator_tag) {if (n > 0) {while (n--) ++it;}else {while (n++) --it;}
}template <class InputIter, class Distance>
void advance(InputIter& it, Distance n, std::input_iterator_tag) {while (n--) {++it;}
}

这几个重载函数的第三个参数就是所谓的 tag,以 std::input_iterator_tag 为例,标准库中的定义大概是这个样子:

struct input_iterator_tag {};

标准库还定义了 `iterator_traits<>` 类模板用于提取迭代器的 tag,对于支持随机访问的迭代器,它的 iterator_category 被特化处理为:

template <class Iter>
struct iterator_traits<Iter> {....using iterator_category = random_access_iterator_tag;
};

可用 iterator_traits<Iter>::iterator_category 提取 Iter 类型迭代器的分类 tag。最终 advance() 的实现大致是这个样子:

template <class Iter, class Distance>
void advance(Iter& it, Distance n) {advance(it, n, typename std::iterator_traits<Iter>::iterator_category{} );
}

3.使用自己的 Tag Dispatching

3.1.使用 type traits 技术

        在介绍 std::enable_if 和 if constexpr 两个主题的时候,我们提到了 `ToString()` 还可以使用 Tag Dispatching 实现,但是没有详细说明。其实 Tag Dispatching 并不是个复杂的技术,那个例子使用 type traits 技术实现分配选择,本篇就借这个主题把这个例子完整解释一下。

        首先要定义 tag,这个例子需要两个 tag 用于区分两种情况:

struct NumTag {};
struct StrTag {};

        理论上说,此时用 `ToString(42, NumTag())` 和 `ToString(std::string("Emma"), StrTag())` 就能区分两个重载函数了,但是我们设计的是针对泛型的函数模板,需要提供一种根据类型提取 tag 的手段。其实就是仿照标准库的样子做一个自己的 traits 类,利用 traits 类的特化版本实现编译期间的 tag 定义:

template <typename T>
struct traits
{typedef NumTag tag;
};template <>
struct traits<std::string>
{typedef StrTag tag;
};

        可以使用 `traits<T>::tag` 提取 T 对应的 tag,针对 `std::string` 提供了一个 `traits<>` 的特化版本,这个版本里的 tag 被定义为 `StrTag`。

        接下来就是实现针对两种 tag 的 `ToString()` 重载版本,为了区分,我们使用 `ToString_impl()` 作为函数名字:

template <typename T>
auto ToString_impl(T t, NumTag)
{return std::to_string(t);
}template <typename T>
auto ToString_impl(T t, StrTag)
{return t;
}

        对于数字类型的数据,用 `std::to_string()` 转换,对于字符串类型的数据,直接返回字符串即可。`ToString_impl()` 函数的第二个参数是哑形参,不需要指定参数名称,编译器会针对这种情况做适当的优化(优化掉这个参数),如果指定参数名字反而会影响编译器的优化判断。

        最后就是提供统一的 `ToString()` 函数,通过 `traits<T>` 提取类型的对应的 tag,让编译器根据 tag 选择正确的重载函数:

template <typename T>
auto ToString(T t)
{return ToString_impl(t, typename traits<T>::tag());
}int main()
{std::cout << ToString(42) << std::endl;std::cout << ToString(std::string("Emma")) << std::endl;
}

3.2.使用 Type_2_Type 技术

        `Type_2_Type` 是一种类型映射技术,常用来将一种普通类型映射为另一种可控类型。Tag Dispatching 也可以借助 `Type_2_Type` 实现类型分派,此时的 tag 也被称为 templated tags。

        首先需要定义一个泛化的 `TypeTag<T>`,用作控制分派的可控类型:

template<typename T>
struct TypeTag {};

        然后修改 `ToString_impl()` 的参数类型,改用我们定义的可控类型做模板参数:

template <typename T>
auto ToString_impl(T t, TypeTag<int>)
{return std::to_string(t);
}template <typename T>
auto ToString_impl(T t, TypeTag<std::string>)
{return t;
}

        最后就是修改 `ToString()` 函数,根据函数参数 t 推导出的类型 T,利用 `TypeTag<T>` 映射为可控类型中的 `TypeTag<int>` 或 `TypeTag<std::string>`,使得编译器可以根据 `TypeTag<T>` 选择正确的重载函数:

template <typename T>
auto ToString(T t)
{return ToString_impl(t, TypeTag<T>());
}

4.Tag Dispatching的使用场景

        编译期需要进行的重载函数分派可以考虑用 Tag Dispatching,运行期间的分派可以考虑 C++ 对象的抽象和分派方式。什么情况适合放在编译期分派呢?对操作或行为需要进行额外控制的场合可以考使用这种编译期进行的 Tag Dispatching,因为这对提高代码运行时的效率非常有用(不需要在运行时对条件进行判断) 。对数据的额外处理就不适合在编译期间决定,因为数据是运行期变化的。

        以下是Tag Dispatching在C++中的一些典型应用场景:

  1. 算法特化(Algorithm Specialization):当算法对于不同的数据类型有不同的最优实现时,可以使用Tag Dispatching来提供特化的版本。例如,对于交换两个元素的操作,对于基本类型可能需要三次拷贝操作,但对于像std::vector这样的容器类型,可以直接使用其成员函数swap来避免拷贝,从而提高效率。
  2. 迭代器类型的优化:在STL(Standard Template Library)中,不同的容器类型具有不同类型的迭代器(如输入迭代器、前向迭代器、双向迭代器和随机访问迭代器)。对于某些算法,根据迭代器的类型选择最优的实现方式可以提高效率。通过使用Tag Dispatching,可以为不同类型的迭代器提供特化的算法实现。
  3. 类型属性的判断:当需要根据类型的某些属性(如是否为整数类型、是否支持某种操作等)来选择不同的行为时,可以使用Tag Dispatching。通过定义与这些属性相关的标签类型,并在函数模板中使用这些标签作为参数,可以在编译时根据类型属性选择正确的实现。
  4. 编译时条件判断:在某些情况下,可能需要在编译时根据某些条件选择不同的函数实现。通过使用if constexpr和Tag Dispatching,可以在编译时根据条件选择并执行相应的函数模板。
  5. 模板元编程:Tag Dispatching在模板元编程中也有广泛应用。通过定义与类型特征相关的标签类型,并在模板元函数中使用这些标签作为参数,可以在编译时根据类型特征执行不同的元编程逻辑。
  6. 类型安全的接口设计:在设计类型安全的接口时,可以使用Tag Dispatching来确保函数只接受特定类型的参数。通过定义与参数类型相关的标签类型,并在函数模板中使用这些标签作为参数,可以在编译时检查参数类型,从而提高代码的类型安全性。

5.总结

        总结来说,Tag Dispatching在C++中主要用于实现泛型算法的优化、迭代器类型的优化、类型属性的判断、编译时条件判断、模板元编程以及类型安全的接口设计等方面。通过使用Tag Dispatching技术,可以根据参数类型或特性在编译时选择最优的实现路径,从而提高代码的性能和可维护性。

推荐阅读:

标签派发

C++之多层 if-else-if 结构优化(二)

C++17之std::invoke: 使用和原理探究(全)

这篇关于C++ 的 Tag Dispatching(标签派发) 惯用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1024610

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

C/C++的编译和链接过程

目录 从源文件生成可执行文件(书中第2章) 1.Preprocessing预处理——预处理器cpp 2.Compilation编译——编译器cll ps:vs中优化选项设置 3.Assembly汇编——汇编器as ps:vs中汇编输出文件设置 4.Linking链接——链接器ld 符号 模块,库 链接过程——链接器 链接过程 1.简单链接的例子 2.链接过程 3.地址和

C++必修:模版的入门到实践

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C++学习 贝蒂的主页:Betty’s blog 1. 泛型编程 首先让我们来思考一个问题,如何实现一个交换函数? void swap(int& x, int& y){int tmp = x;x = y;y = tmp;} 相信大家很快就能写出上面这段代码,但是如果要求这个交换函数支持字符型

C++入门01

1、.h和.cpp 源文件 (.cpp)源文件是C++程序的实际实现代码文件,其中包含了具体的函数和类的定义、实现以及其他相关的代码。主要特点如下:实现代码: 源文件中包含了函数、类的具体实现代码,用于实现程序的功能。编译单元: 源文件通常是一个编译单元,即单独编译的基本单位。每个源文件都会经过编译器的处理,生成对应的目标文件。包含头文件: 源文件可以通过#include指令引入头文件,以使

C++面试八股文:std::deque用过吗?

100编程书屋_孔夫子旧书网 某日二师兄参加XXX科技公司的C++工程师开发岗位第26面: 面试官:deque用过吗? 二师兄:说实话,很少用,基本没用过。 面试官:为什么? 二师兄:因为使用它的场景很少,大部分需要性能、且需要自动扩容的时候使用vector,需要随机插入和删除的时候可以使用list。 面试官:那你知道STL中的stack是如何实现的吗? 二师兄:默认情况下,stack使

剑指offer(C++)--孩子们的游戏(圆圈中最后剩下的数)

题目 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。HF作为牛客的资深元老,自然也准备了一些小游戏。其中,有个游戏是这样的:首先,让小朋友们围成一个大圈。然后,他随机指定一个数m,让编号为0的小朋友开始报数。每次喊到m-1的那个小朋友要出列唱首歌,然后可以在礼品箱中任意的挑选礼物,并且不再回到圈中,从他的下一个小朋友开始,继续0...m-1报数....这样下去

剑指offer(C++)--扑克牌顺子

题目 LL今天心情特别好,因为他去买了一副扑克牌,发现里面居然有2个大王,2个小王(一副牌原本是54张^_^)...他随机从中抽出了5张牌,想测测自己的手气,看看能不能抽到顺子,如果抽到的话,他决定去买体育彩票,嘿嘿!!“红心A,黑桃3,小王,大王,方片5”,“Oh My God!”不是顺子.....LL不高兴了,他想了想,决定大\小 王可以看成任何数字,并且A看作1,J为11,Q为1