0%

C++的模版特化

介绍C++模版的特化与偏特化。

主模版 (1)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#include <iostream>
#include <string>
#include <string_view>
#include <tuple>
#include <utility>

using namespace std;

template<typename T1, typename T2, typename T3>
class MyTemp {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "generalized-type: " << name << std::endl;
}
};

主模版有3个类型参数:T1, T2T3;静态函数(类函数)print_type_info()用于打印出实例化的类型实参,细节可以忽略。它借助于__PRETTY_FUNCTION__;这是函数的签名,里面包含类的信息,即实例化的类型实参。

假如模版没有定义,即:

1
2
template<typename T1, typename T2, typename T3>
class MyTemp;

它也可以实例化,但实例化的类不能构造对象。这是一个细节。

1
2
using T1 = MyTemp<int, char, float>;  //OK
T1 t1; //error: ‘T1 t1’ has incomplete type and cannot be defined

全特化 (2)

特化比较简单,就是针对T1, T2, T3为特定类型提供一个特别的实现。实例化时,若类型参数刚好匹配“特定类型”,就实例化为这个特别实现;否则就用主模版的实现。

1
2
3
4
5
6
7
8
//全特化
template<>
class MyTemp<char, int, float> {
public:
static void print_type_info() {
std::cout << "full-specialized: [char, int, float]" << std::endl;
}
};

实例化:

1
2
3
4
5
using G = MyTemp<std::string, std::string, std::string>; //主模版
using F = MyTemp<char, int, float>; //全特化

G::print_type_info();
F::print_type_info();

MacOS/clang编译器输出(c++17):

1
2
3
4
generalized-type: [T1 = std::string,
T2 = std::string,
T3 = std::string]
full-specialized: [char, int, float]

Linux/gcc编译器输出(c++17):

1
2
3
4
generalized-type: [with T1 = std::basic_string<char>;
T2 = std::basic_string<char>;
T3 = std::basic_string<char>]
full-specialized: [char, int, float]

偏特化 (3)

先看简单形态:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//偏特化1
template<typename T1, typename T2>
class MyTemp<T1, T2, float> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "partial-specialized-1: " << name << std::endl;
}
};

//偏特化2
template<typename T1>
class MyTemp<T1, int, float> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "partial-specialized-2: " << name << std::endl;
}
};

这里提供了2个偏特化:

  • 偏特化1:只固定第3个类型参数为float
  • 偏特化2:固定第2个类型参数为int,固定第3个类型参数为float

那么,实例化的的时候,如匹配偏特化2必然匹配偏特化1(例如MyTemp<string, int, float>),最终选择哪个呢?当然是选择匹配最多的,遵循最长匹配原则,即偏特化2;

1
2
3
4
using P1 = MyTemp<double, double, float>; //偏特化1
using P2 = MyTemp<double, int, float>; //偏特化2
P1::print_type_info();
P2::print_type_info();

MacOS/clang编译器输出(c++17):

1
2
partial-specialized-1: [T1 = double, T2 = double, T3 = float]
partial-specialized-2: [T1 = double, T2 = int, T3 = float]

Linux/gcc编译器输出(c++17):

1
2
partial-specialized-1: [with T1 = double; T2 = double]
partial-specialized-2: [with T1 = double]

似乎可以总结:特化就是把template<type-arg-list-1> class MyTemp<type-arg-list-2>里的“type-arg-list-1”缩短,把“type-arg-list-2”中形参固化成真实类型。当“type-arg-list-1”为空的时候,“type-arg-list-2”里的形参全部固化成真实类型,这时就变成“full-specialization”了(见第2节)。

但实际上,固定“type-arg-list-2”的时候,不一定非要缩短“type-arg-list-1”!特化不是从“type-arg-list-1”中移除,在“type-arg-list-2”中固化的过程!看下面的例子;

1
2
3
4
5
6
7
8
9
10
//偏特化3
template<typename U, typename V, typename W, int a, int b>
class MyTemp<std::tuple<U,V,W>, char[a], double[b]> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "partial-specialized-3: " << name << std::endl;
}
};

当“type-arg-list-2”固定的时候,“type-arg-list-1”不但没有缩短,反而曾长了!特化/偏特化确实是固化“type-arg-list-2”中的类型参数,但不是为了缩短“type-arg-list-1”!实际上,“type-arg-list-1”为“type-arg-list-2”提供类型信息:即“type-arg-list-2”中的一切未固定信息,例如类型参数,int参数等,都要由“type-arg-list-1”提供

  • 对于偏特化1和偏特化2的简单情形,在“type-arg-list-2”固化的过程中,需要的类型信息减少了,所以“type-arg-list-1”缩短了。甚至在“full-specialization”中“type-arg-list-1”缩为0,因为“type-arg-list-2”不需要任何类型信息;
  • 但对于偏特化3,为了固化“type-arg-list-2”:T1固化为std::tuple<U,V,W>需要3个类型参数U, VWT2固化为char[a]T3固化为double[b]还需要2个int类型的型参ab;这些都需要“type-arg-list-1”来提供。

在为一个模版提供特化/偏特化的时候也是这个思路:先想着去固定T1, T2T3;当这些类型能全部固定的时候,就是全特化,“type-arg-list-1”为空;当这些类型中有不可以固定的东西,无论是类型参数,还是int这样的变量参数,都放到“type-arg-list-1”中,这就是偏特化

那么如何实例化偏特化3呢?

1
2
using P3 = MyTemp<std::tuple<int, float, double>, char[100], double[200]>;
P3::print_type_info();

MacOS/clang编译器输出(c++17):

1
partial-specialized-3: [T1 = std::tuple<int, float, double>, T2 = char[100], T3 = double[200]]

Linux/gcc编译器输出(c++17):

1
partial-specialized-3: [with U = int; V = float; W = double; int a = 100; int b = 200]

Linux/gcc的输出中明确地推导出了U, VW以及a, b;不过从MacOS/clang的输出,可以更清楚地看到T1, T2T3的实参。

注意:

  • 实例化时不是要提供U, VW以及a, b,而是要提供与std::tuple<U,V,W>, char[a], double[b]匹配的T1, T2T3
  • 编译器会推导出U, VW以及a, b可以利用编译器的推导能力来提取基础类型;例如在偏特化3中添加using type_u = U;,那么在上面的实例化中,就从std::tuple<int, float, double>中提取出type_u = int;就是说,T1又是一个模版类,模版类有自己的类型参数X, Y, Z(这时,X, Y, Z要放在“type-arg-list-1”中),编译器会推导出它们的实际类型,所以using type_x = X;就可以得到真实的X类型;

另外,似乎3个类型参数都固化了,看上去像一个“full-specialization”?其实不是的,“full-specialization”不是那么定义的,是看模版还有没有自由度。显然这里还有自由度,std::tuple<U,V,W>, char[a], double[b]中的U, VW以及a, b都可以随意选择。

最后,再看一个更复杂的例子:

1
2
3
4
5
6
7
8
9
10
//偏特化4
template<typename U, typename V, typename... W, size_t... a, size_t... b>
class MyTemp<std::tuple<U, V, W...>, std::index_sequence<a...>, std::index_sequence<b...>> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "partial-specialized-4: " << name << std::endl;
}
};

这里甚至引入了“参数包”:固化“type-arg-list-2”中的T1 = std::tuple<U, V, W...>T2 = std::index_sequence<a...>, T3 = std::index_sequence<b...>需要“type-arg-list-1”提供U, V, W...以及a...b...

实例化:

1
2
using P4 = MyTemp<std::tuple<int, float, double, char, std::string>, std::index_sequence<1,2,3>, std::index_sequence<4,5,6,7,8,9>>;
P4::print_type_info();

MacOS/clang编译器输出(c++17):

1
2
3
partial-specialized-4: [T1 = std::tuple<int, float, double, char, std::string>,
T2 = std::integer_sequence<unsigned long, 1, 2, 3>,
T3 = std::integer_sequence<unsigned long, 4, 5, 6, 7, 8, 9>]

Linux/gcc编译器输出(c++17):

1
2
3
4
5
partial-specialized-4: [with U = int; 
V = float;
W = {double, char, std::basic_string<char, std::char_traits<char>, std::allocator<char> >};
long unsigned int ...a = {1, 2, 3};
long unsigned int ...b = {4, 5, 6, 7, 8, 9}]

同样,Linux/gcc的输出中明确地推导出U, V, W...以及a..., b...;不过MacOS/clang的输出中,可以更清楚地看到T1, T2T3的实参。

特化模版会继承主模版的基类吗 (4)

先说答案:不同编译器行为不同!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
#include <iostream>
#include <string>
#include <string_view>
#include <tuple>
#include <utility>

using namespace std;

template<typename T>
class B {
public:
static_assert(false, "B cannot be instantiated");
};

template<typename T>
class C : B<T> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "generalized-type: " << name << std::endl;
}
};

template<typename T>
class C<std::tuple<T>> {
public:
static void print_type_info() {
std::string_view name = __PRETTY_FUNCTION__;
name.remove_prefix(name.find("print_type_info() ") + 18);
std::cout << "partial-specialized: " << name << std::endl;
}
};

因为static_assert(false, ...),所以B不能被实例化!实例化一个C的类时,若命中主模版则编译失败,这个对于MacOS/clang和Linux/gcc行为是一样的!

1
2
using C1 = C<int>;
C1::print_type_info(); //error: static assertion failed: B cannot be instantiated

捣鬼的是命中偏特化的情形:

1
2
using C2 = C<std::tuple<int>>;
C2::print_type_info();

MacOS/clang编译器输出(c++17):

1
partial-specialized: [T = std::tuple<int>]

Linux/gcc编译器则编译出错,输出(c++17):

1
2
error: static assertion failed: B cannot be instantiated
static_assert(false, "B cannot be instantiated");

那是不是Linux/gcc命中了主模版呢(匹配规则不同)?不是的,把static_assert注释掉,可以发现,Linux/gcc也是命中偏特化。所不同的是,它也会实例化主模版的基类!

小节 (5)

总结C++模版的特化以及偏特化,特别是偏特化中的复杂情况。

写的不错,有赏!