本 Wiki 开启了 HTTPS。但由于同 IP 的 Blog 也开启了 HTTPS,因此本站必须要支持 SNI 的浏览器才能浏览。为了兼容一部分浏览器,本站保留了 HTTP 作为兼容。如果您的浏览器支持 SNI,请尽量通过 HTTPS 访问本站,谢谢!
这里会显示出您选择的修订版和当前版本之间的差别。
两侧同时换到之前的修订记录前一修订版后一修订版 | 前一修订版 | ||
cs:programming:cpp:cpp_primer:16_template [2024/01/14 13:46] – 移除 - 外部编辑 (未知日期) 127.0.0.1 | cs:programming:cpp:cpp_primer:16_template [2024/11/17 04:38] (当前版本) – [使用 std::forward] codinghare | ||
---|---|---|---|
行 1: | 行 1: | ||
+ | | ||
+ | C++ Primer 笔记 第十六章\\ | ||
+ | ---- | ||
+ | ====模板的定义==== | ||
+ | 模板(// | ||
+ | ===函数模板的定义=== | ||
+ | 模板的定义由以下两部分组成: | ||
+ | * 关键词 '' | ||
+ | * **模板参数列表**(// | ||
+ | 一个模板函数的实现: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | int compare(const T &v1, const T &v2) | ||
+ | { | ||
+ | if (v1 < v2) return -1; | ||
+ | if (v2 < v1) return 1; | ||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | 模板参数列表中的 argument 被称为 tempalte argument;这些参数会与函数的参数绑定,而作为类型的 '' | ||
+ | ===函数模板的实例化=== | ||
+ | 函数模板再被调用的时候,会根据绑定参数的类型来推断出模板类型的具体类型;这个过程被称为**模板的实例化**(// | ||
+ | ==模板类型的 parameter== | ||
+ | 以之前的 '' | ||
+ | * 函数的**返回值**类型 | ||
+ | * 函数的**参数**类型 | ||
+ | * 函数内部的**本地变量**类型 | ||
+ | <code cpp> | ||
+ | // ok: same type used for the return type and parameter | ||
+ | template < | ||
+ | { | ||
+ | T tmp = *p; // tmp will have the type to which p points | ||
+ | // ... | ||
+ | return tmp; | ||
+ | } | ||
+ | </ | ||
+ | 表示类型的 模板类型 parameter 需要被关键字 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | </ | ||
+ | 两个关键字是等价的,但由于 '' | ||
+ | ==非类型的模板 parameter== | ||
+ | 除了表示类型的模板 parameter 以外,还有一种表示**值**的模板 parameter。这种模板参数被成为**非类型** parameter(// | ||
+ | 当模板实例化的时候,非类型的 parameter 会被替换为函数获取的对应值(或是由编译器推断决定)。由于模板的实例化实在编译期进行,因此该值必须是**常量表达式**。比如下方的例子,我们可以利用非类型的 parameter 对数组的长度进行替换: | ||
+ | <code cpp> | ||
+ | template< | ||
+ | int compare(const char (& | ||
+ | { | ||
+ | return strcmp(p1, p2); | ||
+ | } | ||
+ | </ | ||
+ | 当调用以下命令时,'' | ||
+ | <code cpp> | ||
+ | compare(" | ||
+ | </ | ||
+ | 非类型 parameter 可以表示以下类型: | ||
+ | * intergral type | ||
+ | * 指针 | ||
+ | * 对象或者函数左值引用 | ||
+ | 指针和引用绑定的 argument 必须拥有**静态生命周期**(// | ||
+ | <WRAP center round info 100%> | ||
+ | 绑定 //nullptr// 的指针可以作为非类型参数。 | ||
+ | </ | ||
+ | <WRAP center round box 100%> | ||
+ | 一切都是常量表达式的要求。 | ||
+ | </ | ||
+ | ==inline 和 constexpr 类型的函数模板== | ||
+ | '' | ||
+ | <code cpp> | ||
+ | // ok: inline specifier follows the template parameter list | ||
+ | template < | ||
+ | // error: incorrect placement of the inline specifier | ||
+ | inline template < | ||
+ | </ | ||
+ | ==模板需要最小化参数类型的影响== | ||
+ | 本章开头的例子 '' | ||
+ | - 首先,我们可以通过统一运算符来减少模板对类型的依赖。比如该函数的大小比较中,只使用了 ''<'' | ||
+ | - 将函数参数设置为 '' | ||
+ | ==模板的编译== | ||
+ | 模板下的代码生成与普通函数不太一样: | ||
+ | * 模板在定义可见的时候不会生成代码;只有在实例化的过程中会生成代码。 | ||
+ | * 编译器在模板在实例化的过程中需要**其声明与定义都可见**(类似 // | ||
+ | <WRAP center round box 100%> | ||
+ | 关键概念: | ||
+ | * 模板的提供者需要保证模板对实例化时的参数(带具体类型的参数)没有依赖 | ||
+ | * 模板的提供者需要保证实例化时,模板的声明和定义是完整可见的 | ||
+ | * 而具体的,需要被实例化的类型参数的正确性,由使用者保证 | ||
+ | 实际操作中,由编写者提供带模板实现的头文件,使用者包含该头文件并提供对应的实例化信息。 | ||
+ | </ | ||
+ | <WRAP center round box 100%> | ||
+ | [[https:// | ||
+ | </ | ||
+ | ==编译错误几乎都处于实例化阶段== | ||
+ | 模板的编译存在三个阶段: | ||
+ | - 模板的自我编译阶段,这个阶段只会检查语法错误 | ||
+ | - 模板被初次使用时,这个阶段会检查调用者的参数数量,以及参数是否具有相同的类型 | ||
+ | - 模板的实例化阶段,类型相关的错误只会在这个阶段被发现(可能会在链接期报错) | ||
+ | 比如本章开头的 '' | ||
+ | <code cpp> | ||
+ | if (v1 < v2) return -1; // requires < on objects of type T | ||
+ | </ | ||
+ | 如果我们传入没有定义 ''<'' | ||
+ | ===类模板的定义与实例化=== | ||
+ | 类模板(// | ||
+ | ==类模板的定义== | ||
+ | 类模板的定义分几个点: | ||
+ | * 类模板名之前需要加上模板关键词与模板参数列表,比如 | ||
+ | * 类中所有由模板指定的具体类型,都需要用对应的模板参数替换掉,比如 '' | ||
+ | 以 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | public: | ||
+ | typedef T value_type; | ||
+ | typedef typename std:: | ||
+ | // constructors | ||
+ | Blob(); | ||
+ | Blob(std:: | ||
+ | // number of elements in the Blob | ||
+ | size_type size() const { return data-> | ||
+ | bool empty() const { return data-> | ||
+ | // add and remove elements | ||
+ | void push_back(const T &t) {data-> | ||
+ | // move version; see § 13.6.3 (p. 548) | ||
+ | void push_back(T && | ||
+ | void pop_back(); | ||
+ | // element access | ||
+ | T& back(); | ||
+ | T& operator[](size_type i); // defined in § 14.5 (p. 566) | ||
+ | private: | ||
+ | std:: | ||
+ | // throws msg if data[i] isn't valid | ||
+ | void check(size_type i, const std::string &msg) const; | ||
+ | }; | ||
+ | </ | ||
+ | ==类模板的实例化== | ||
+ | 类模板的实例化需要提供额外的信息。这些信息被称为**显式模板实参** (//explicit template arguments// | ||
+ | <code cpp> | ||
+ | Blob< | ||
+ | </ | ||
+ | 每个类模板的实例化创建的都是一个**独立的类类型**,相互之间没有任何联系。 | ||
+ | ==在模板的作用域类引用模板类型== | ||
+ | * 类模板名不是类型,其实例化后的类才能表示类型;这种类型被成为**实例化类型** | ||
+ | * 实例化类型由**类模板名**和**模板 parameter**(注意**不是**具体的 //template argument// | ||
+ | 比如: | ||
+ | <code cpp> | ||
+ | //data definition | ||
+ | std:: | ||
+ | //if instantiated by Blob< | ||
+ | //data type | ||
+ | shared_ptr< | ||
+ | </ | ||
+ | ==定义模板类的成员函数== | ||
+ | * 模板类的成员函数会受到模板类实例化的影响。其 template parameter 与模板类一致 | ||
+ | * 定义于模板类外部的成员函数,其所属必须带上 template parameter list | ||
+ | * 成员函数所属的类也必须带上 template parameter | ||
+ | <code cpp> | ||
+ | //template parameter list the first | ||
+ | template < | ||
+ | //which class the member belongs the second | ||
+ | //if return type is the class type, it should include the template parameter as well | ||
+ | return-type | ||
+ | Blob< | ||
+ | </ | ||
+ | ==Blob 的相关实现及注意要点== | ||
+ | 普通成员中,主要替换的是所属的类,以及返回类型。注意模板关键字与模板参数列表都是必不可少的: | ||
+ | <code cpp> | ||
+ | /* check */ | ||
+ | template < | ||
+ | void Blob< | ||
+ | { | ||
+ | if (i >= data-> | ||
+ | throw std:: | ||
+ | } | ||
+ | |||
+ | /* back */ | ||
+ | template < | ||
+ | T& Blob< | ||
+ | { | ||
+ | check(0, "back on empty Blob" | ||
+ | return data-> | ||
+ | } | ||
+ | |||
+ | /* subscript */ | ||
+ | template < | ||
+ | T& Blob< | ||
+ | { | ||
+ | // if i is too big, check will throw, preventing access to a nonexistent element | ||
+ | check(i, " | ||
+ | return (*data)[i]; | ||
+ | } | ||
+ | |||
+ | /* pop_back */ | ||
+ | template < | ||
+ | { | ||
+ | check(0, " | ||
+ | data-> | ||
+ | } | ||
+ | </ | ||
+ | 构造函数中类似。如果构造函数接受任何模板类型的 parameter,需要将类模板的 template parameter 对其类型进行替换: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | Blob< | ||
+ | |||
+ | template < | ||
+ | Blob< | ||
+ | data(std:: | ||
+ | </ | ||
+ | ==类模板成员函数的实例化== | ||
+ | <WRAP center round important 100%> | ||
+ | 类模板成员函数只有在被**使用的情况下才会实例化**。 | ||
+ | </ | ||
+ | ==在模板类中使用该模板类型== | ||
+ | 模板类类型在**类中**可以**省略** template argument。比如下例中的 '' | ||
+ | <code cpp> | ||
+ | // BlobPtr throws an exception on attempts to access a nonexistent element | ||
+ | template < | ||
+ | public: | ||
+ | BlobPtr(): curr(0) { } | ||
+ | BlobPtr(Blob< | ||
+ | wptr(a.data), | ||
+ | T& operator*() const | ||
+ | { auto p = check(curr, " | ||
+ | return (*p)[curr]; | ||
+ | } | ||
+ | // increment and decrement | ||
+ | BlobPtr& | ||
+ | BlobPtr& | ||
+ | private: | ||
+ | // check returns a shared_ptr to the vector if the check succeeds | ||
+ | std:: | ||
+ | check(std:: | ||
+ | // store a weak_ptr, which means the underlying vector might be destroyed | ||
+ | std:: | ||
+ | std::size_t curr; // current position within the array | ||
+ | }; | ||
+ | </ | ||
+ | 编译器默认所有模板类中的模板类型都拥有 template argument。 | ||
+ | ==在模板类外使用该模板类型== | ||
+ | 如果在模板类外部使用模板类类型,那该类型必须是完全体,也就是**模板名 + 模板参数列表 + 模板类所属 + 函数定义**的形式,一个都不能少。比如下方的例子 | ||
+ | <code cpp> | ||
+ | // postfix: increment/ | ||
+ | template < | ||
+ | BlobPtr< | ||
+ | { | ||
+ | // no check needed here; the call to prefix increment will do the check | ||
+ | BlobPtr ret = *this; | ||
+ | ++*this; | ||
+ | return ret; // return the saved state | ||
+ | } | ||
+ | </ | ||
+ | 需要注意的是,该函数定义内部的作用域属于模板类内部,因此 '' | ||
+ | ===类模板与友元=== | ||
+ | 类模板与友元的关系可以大致定义为以下几类: | ||
+ | * 一对一的关系(// | ||
+ | * 特定的一对一关系 | ||
+ | * 一对多的关系 | ||
+ | * 多对一的关系 | ||
+ | * 多对多的关系 | ||
+ | 这些关系的对象可以是模板与具体(函数、类)的关系,也可以是模板与模板之间的关系。 | ||
+ | ==一对一的关系== | ||
+ | 需要提供模板的**前置声明**。 | ||
+ | \\ \\ | ||
+ | 当友元关系为一对一的关系(// | ||
+ | |||
+ | <code cpp> | ||
+ | // forward declarations needed for friend declarations in Blob | ||
+ | template < | ||
+ | template < | ||
+ | template < | ||
+ | |||
+ | template < | ||
+ | { | ||
+ | // each instantiation of Blob grants access to the version of | ||
+ | // BlobPtr and the equality operator instantiated with the same type | ||
+ | | ||
+ | // class friend | ||
+ | friend class BlobPtr< | ||
+ | | ||
+ | //function friend | ||
+ | friend bool operator==< | ||
+ | // ... | ||
+ | }; | ||
+ | </ | ||
+ | 重点在于友元模板类与函数模板之后的那个 template argument ''< | ||
+ | <code cpp> | ||
+ | friend class BlobPtr< | ||
+ | friend bool operator==< | ||
+ | </ | ||
+ | |||
+ | 当这个 ''< | ||
+ | <code cpp> | ||
+ | // BlobPtr< | ||
+ | // operator==< | ||
+ | Blob< | ||
+ | </ | ||
+ | |||
+ | ==特定的一对一关系: | ||
+ | 需要提供模板的前置声明。 | ||
+ | \\ \\ | ||
+ | 除了上述的一种一对一关系以外,如果模板类以某个具体类作为类型进行实例化,那么实例化后的类型将可以以友元的方式访问具体类: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | class C { // C is an ordinary, nontemplate class | ||
+ | friend class Pal< | ||
+ | }; | ||
+ | </ | ||
+ | 上例中,'' | ||
+ | ==多对一的关系:类模板的所有实例与对应具体类== | ||
+ | **不需要**提供任何形式的前置声明。 | ||
+ | \\ \\ | ||
+ | 该关系描述的是类模板的所有实例,都是某一个具体类的友元的关系: | ||
+ | <code cpp> | ||
+ | class C { // C is an ordinary, nontemplate class | ||
+ | // all instances of Pal2 are friends to C; | ||
+ | // no forward declaration required when we befriend all instantiations | ||
+ | template < | ||
+ | }; | ||
+ | </ | ||
+ | 以上面的例子来讲,如果模板类 '' | ||
+ | |||
+ | ==一对多的关系:具体类与对应类模板的所有实例== | ||
+ | **不需要**提供任何形式的前置声明。 | ||
+ | \\ \\ | ||
+ | 该关系的描述的是:当某个具体类声明了与类模板的友元关系,那么该具体类可以以友元的方式访问类模板的所有实例化类型对象: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | { | ||
+ | // Pal3 is a nontemplate class that is a friend of every instance of C2 | ||
+ | friend class Pal3; // prior declaration for Pal3 not needed | ||
+ | }; | ||
+ | </ | ||
+ | 上面的例子中,具体类 '' | ||
+ | ==多对多的关系:类模板与类模板== | ||
+ | 不需要提供任何形式的前置声明。 | ||
+ | \\ \\ | ||
+ | 该关系描述的是,在类模板中以非一对一形式声明的友元关系,可以使对应的**类模板的所有实例**以友元的方式访问**目标类模板的所有实例**。需要注意的是,这种情况下声明的友元类模板,必须拥有独立的 template argument,否则编译时会报 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | { | ||
+ | // all instances of Pal2 are friends of each instance of C2; no prior declaration needed | ||
+ | template < | ||
+ | } | ||
+ | </ | ||
+ | 上例中, 所有 '' | ||
+ | ==以模板参数类型作为友元== | ||
+ | C++11 中提供了以模板参数类型作为友元的功能:如果类模板以某种类型实例化,且类模板中声明了对该类型的友元关系;那么所有以该种类型定义的**对象,函数**(甚至 built-in type)都可以以友元的方式访问类模板的该种类型实例: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | friend Type; // grants access to the type used to instantiate Bar | ||
+ | // ... | ||
+ | }; | ||
+ | </ | ||
+ | 举例来说,如果 '' | ||
+ | <code cpp> | ||
+ | template< | ||
+ | class Bar { | ||
+ | friend T; | ||
+ | protected: | ||
+ | int val= 100; | ||
+ | }; | ||
+ | |||
+ | class Foo { | ||
+ | public: | ||
+ | void print_bar(Bar< | ||
+ | }; | ||
+ | |||
+ | int main(int argc, char const *argv[]) | ||
+ | { | ||
+ | Bar< | ||
+ | Foo foo; | ||
+ | foo.print_bar(bar); | ||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | ===类模板的其他功能=== | ||
+ | ==Template Type Aliases== | ||
+ | 早期的 C++ 可以做类模板实例的 alias: | ||
+ | <code cpp> | ||
+ | typedef Blob< | ||
+ | </ | ||
+ | C++ 11 中可以对类模板直接进行 alias: | ||
+ | <code cpp> | ||
+ | template< | ||
+ | twin< | ||
+ | </ | ||
+ | 对于存在多个参数的情况,可以单独指定某个参数作为 alias: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | partNo< | ||
+ | partNo< | ||
+ | partNo< | ||
+ | </ | ||
+ | ==类模板的静态成员== | ||
+ | 类模板的静态成员以**实例类型为单位**:所有类型相同的实例共享一个静态成员,但不同类型的实例之间存在不同的静态成员: | ||
+ | <code cpp> | ||
+ | // instantiates static members Foo< | ||
+ | Foo< | ||
+ | // all three objects share the same Foo< | ||
+ | Foo< | ||
+ | </ | ||
+ | 因为静态成员以实例类型为单位,因此定义时需要加上 template argument: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | size_t Foo< | ||
+ | </ | ||
+ | 与之对应的是,访问静态成员时也需要指定实例化的类型: | ||
+ | <code cpp> | ||
+ | Foo< | ||
+ | // and the static data member ctr | ||
+ | auto ct = Foo< | ||
+ | ct = fi.count(); | ||
+ | ct = Foo:: | ||
+ | </ | ||
+ | ===模板参数=== | ||
+ | 模板参数**名字**没有内在的意义;'' | ||
+ | ==模板参数和作用域== | ||
+ | * 模板参数的名字会**隐藏**任意**外部定义的同名** | ||
+ | * 被用于模板参数的名字**不能被内部变量再次使用** | ||
+ | <code cpp> | ||
+ | typedef double A; | ||
+ | template < | ||
+ | { | ||
+ | A tmp = a; // tmp has same type as the template parameter A, not double | ||
+ | double B; // error: redeclares template parameter B | ||
+ | } | ||
+ | </ | ||
+ | 上例中,tmp 对应的 '' | ||
+ | * 模板参数的名称是**唯一的** | ||
+ | <code cpp> | ||
+ | // error: illegal reuse of template parameter name V | ||
+ | template < | ||
+ | </ | ||
+ | ==模板的声明== | ||
+ | * 模板的声明需要包含模板参数: | ||
+ | <code cpp> | ||
+ | // declares but does not define compare and Blob | ||
+ | template < | ||
+ | template < | ||
+ | </ | ||
+ | * 声明中的模板参数名不要求与定义中相同,但数量与类型要求一致: | ||
+ | <code cpp> | ||
+ | // all three uses of calc refer to the same function template | ||
+ | template < | ||
+ | template < | ||
+ | // definition of the template | ||
+ | template < | ||
+ | Type calc(const Type& a, const Type& b) { /* . . . */ } | ||
+ | </ | ||
+ | <WRAP center round info 100%> | ||
+ | 所有模板的声明应该放置于使用这些模板的代码所在文件的头部。 | ||
+ | </ | ||
+ | ==使用模板类类型的成员== | ||
+ | 由于编译器在模板实例化之前无法确定类模板的具体类型,在下例这种情况中,编译器无法确定 '' | ||
+ | <code cpp> | ||
+ | T:: | ||
+ | </ | ||
+ | * 当 '' | ||
+ | * 当 '' | ||
+ | 默认情况会将 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | typename T:: | ||
+ | { | ||
+ | if (!c.empty()) | ||
+ | return c.back(); | ||
+ | else | ||
+ | return typename T:: | ||
+ | } | ||
+ | </ | ||
+ | <WRAP center round info 100%> | ||
+ | 这种情况下不能使用 //class// 代替 // | ||
+ | </ | ||
+ | ==默认的 template argument== | ||
+ | C++ 11 中允许在函数 / 类模板中使用默认的 template argument: | ||
+ | <code cpp> | ||
+ | // compare has a default template argument, less< | ||
+ | // and a default function argument, F() | ||
+ | template < | ||
+ | int compare(const T &v1, const T &v2, F f = F()) | ||
+ | { | ||
+ | if (f(v1, v2)) return -1; | ||
+ | if (f(v2, v1)) return 1; | ||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | 上例中: | ||
+ | * 模板参数 '' | ||
+ | * 在没有重新指定giant参数时,被使用的 '' | ||
+ | 调用时如果指定了其他的 callable object,则使用该对象作为比较策略: | ||
+ | <code cpp> | ||
+ | bool i = compare(0, 42); // uses less; i is -1 | ||
+ | // uses compareIsbn, | ||
+ | Sales_data item1(cin), item2(cin); | ||
+ | bool j = compare(item1, | ||
+ | </ | ||
+ | 如果希望只使用默认的模板参数,那么三角括号 ''<>'' | ||
+ | <code cpp> | ||
+ | template <class T = int> class Numbers { // by default T is int | ||
+ | public: | ||
+ | Numbers(T v = 0): val(v) { } | ||
+ | // various operations on numbers | ||
+ | private: | ||
+ | T val; | ||
+ | }; | ||
+ | Numbers< | ||
+ | Numbers<> | ||
+ | </ | ||
+ | ===成员模板=== | ||
+ | 成员模板(// | ||
+ | ==普通类的成员模板== | ||
+ | * 成员模板具有**独立**的模板参数列表 | ||
+ | <code cpp> | ||
+ | // function-object class that calls delete on a given pointer | ||
+ | class DebugDelete { | ||
+ | public: | ||
+ | DebugDelete(std:: | ||
+ | // as with any function template, the type of T is deduced by the compiler | ||
+ | template < | ||
+ | { os << " | ||
+ | private: | ||
+ | std:: | ||
+ | }; | ||
+ | </ | ||
+ | 此处重载运算符 '' | ||
+ | <code cpp> | ||
+ | /* bulit-in variable */ | ||
+ | double* p = new double; | ||
+ | DebugDelete d; // an object that can act like a delete expression | ||
+ | d(p); // calls DebugDelete:: | ||
+ | int* ip = new int; | ||
+ | // calls operator()(int*) on a temporary DebugDelete object | ||
+ | DebugDelete()(ip); | ||
+ | |||
+ | /* unique_ptr */ | ||
+ | // destroying the the object to which p points | ||
+ | // instantiates DebugDelete:: | ||
+ | unique_ptr< | ||
+ | // destroying the the object to which sp points | ||
+ | // instantiates DebugDelete:: | ||
+ | unique_ptr< | ||
+ | </ | ||
+ | <WRAP center round box 100%> | ||
+ | 智能指针的构造函数使用方法可以 参考 cpp reference。此处的 //deleter// 以**临时无名**对象 '' | ||
+ | </ | ||
+ | ==类模板的成员模板== | ||
+ | * 类模板的模板参数与成员模板参数是**独立**于彼此的 | ||
+ | <code cpp> | ||
+ | template < | ||
+ | template < | ||
+ | // ... | ||
+ | }; | ||
+ | </ | ||
+ | 这里的 '' | ||
+ | * **成员模板是函数模板** | ||
+ | * 成员模板需要有**独立的模板参数列表**,用于函数参数 | ||
+ | * 类模板的模板参数列表在前,成员模板的列表在后 | ||
+ | <code cpp> | ||
+ | template < | ||
+ | template < | ||
+ | Blob< | ||
+ | data(std:: | ||
+ | </ | ||
+ | 上面的例子展示了成员模板的类外定义的方法,可以看出需要同时写出类模板与成员模板的模板参数列表。 | ||
+ | ==成员模板的实例化== | ||
+ | * 成员模板的实例化必须同时提供类模板与成员模板的 template arguments | ||
+ | <code cpp> | ||
+ | int ia[] = {0, | ||
+ | vector< | ||
+ | list< | ||
+ | // instantiates the Blob< | ||
+ | // and the Blob< | ||
+ | Blob< | ||
+ | // instantiates the Blob< | ||
+ | // two vector< | ||
+ | Blob< | ||
+ | // instantiates the Blob< | ||
+ | // constructor that has two (list< | ||
+ | Blob< | ||
+ | </ | ||
+ | ===显式实例化=== | ||
+ | 由于模板本身并不是类型,因此模板的实例化需要两个部分: | ||
+ | * 模板的声明 | ||
+ | * 模板的定义 | ||
+ | 又因为模板实例化是在编译期完成的,因此通常情况下,模板的实例化要求声明与定义同时可见;这也是为什么模板通常把声明和定义都放到头文件中。但这样会带来两个问题: | ||
+ | * 没有办法将声明与实现分离 | ||
+ | * 如果存在多个文件,编译器在无法得知哪些文件需要使用实例化的情况下,会对**所有文件进行实例化**。这样会带来额外的开销。 | ||
+ | <WRAP center round box 100%> | ||
+ | 编译器无法得知所有文件对实例化的需求(也就是说,'' | ||
+ | [[https:// | ||
+ | </ | ||
+ | |||
+ | C++ 11 中提供了**显式实例化**(// | ||
+ | * 哪些**类型**需要实例化 | ||
+ | * 哪些**文件**需要实例化 | ||
+ | C++ 通过**分离实例化的声明和定义**来达到这样的效果: | ||
+ | <code cpp> | ||
+ | extern template declaration; | ||
+ | template declaration; | ||
+ | </ | ||
+ | 这两行实际上代替了之前普通实例化的定义部分: | ||
+ | * 带有 '' | ||
+ | * 第二行则是实例化的部分。这部分会放到需要进行实例化的文件中。当这部分对编译器可见时,编译器才会对模板进行实例化。 | ||
+ | |||
+ | ==显式实例化的使用== | ||
+ | 需要说明的是,如果是单纯的显式实例化,只需要在使用的地方对模板进行定义即可,并不会用到 '' | ||
+ | <code cpp> | ||
+ | // Foo.h | ||
+ | // no implementation | ||
+ | template < | ||
+ | |||
+ | //Foo.cpp | ||
+ | #include " | ||
+ | // implementation of Foo's methods | ||
+ | |||
+ | // explicit instantiations | ||
+ | template class Foo< | ||
+ | template class Foo< | ||
+ | // You will only be able to use Foo with int or float | ||
+ | </ | ||
+ | '' | ||
+ | 而 '' | ||
+ | \\ \\ < | ||
+ | \\ \\ | ||
+ | |||
+ | 这种情况下就需要**抑制实现文件里的实例化**,因为我们只希望在使用的地方实例化。因此: | ||
+ | * 实现的文件里需要进行 '' | ||
+ | * 使用的文件里进行模板的定义,说明我们需要在这里使用模板的实例 | ||
+ | 下面是一个具体的三文件结构例子: | ||
+ | <code cpp> | ||
+ | /* template.h */ | ||
+ | |||
+ | #ifndef TEMPLATE_H | ||
+ | #define TEMPLATE_H | ||
+ | |||
+ | //template declaration | ||
+ | template < | ||
+ | |||
+ | #endif | ||
+ | |||
+ | /* template.cc */ | ||
+ | |||
+ | #include < | ||
+ | #include " | ||
+ | |||
+ | template < | ||
+ | { | ||
+ | public: | ||
+ | void set(T val) { val = x; } | ||
+ | void print() { std::cout << x; } | ||
+ | private: | ||
+ | T x{}; | ||
+ | }; | ||
+ | |||
+ | //extern declaration, | ||
+ | extern template class Test< | ||
+ | |||
+ | /* main.cc */ | ||
+ | |||
+ | #include " | ||
+ | #include " | ||
+ | |||
+ | //template definition | ||
+ | template class Test< | ||
+ | |||
+ | int main(int argc, char const *argv[]) | ||
+ | { | ||
+ | Test< | ||
+ | it.set(1); | ||
+ | it.print(); | ||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | 除此之外,函数模板也可以通过这种方式进行显示实例化。以上面的成员函数 '' | ||
+ | <code cpp> | ||
+ | /* template.h */ | ||
+ | template < | ||
+ | { | ||
+ | public: | ||
+ | void set(T val); // declaration of set | ||
+ | //....other implementation | ||
+ | }; | ||
+ | |||
+ | /* template.cc */ | ||
+ | |||
+ | // | ||
+ | template < | ||
+ | void | ||
+ | Test< | ||
+ | { | ||
+ | x = val; | ||
+ | } | ||
+ | |||
+ | //extern declaration | ||
+ | extern template void Test< | ||
+ | |||
+ | /* main.cc */ | ||
+ | |||
+ | //template definition | ||
+ | template void Test< | ||
+ | |||
+ | // | ||
+ | </ | ||
+ | ==一些注意事项== | ||
+ | * '' | ||
+ | * 实例化的内容存储于 '' | ||
+ | * 类模板的实例化会**同时实例化**所有成员,即便成员没有被使用。显式实例化也可以应用于单个成员(见上例) | ||
+ | * 显式实例化的定义类型需要满足成员函数的要求 | ||
+ | ===效率与灵活性=== | ||
+ | 在实际的设计过程中,我们可以通过模板类型的特性来帮助我们区分不同的设计需求。比较典型的是智能指针的 deleter: | ||
+ | * // | ||
+ | * // | ||
+ | 根据这两种不同的需求,我们会发现: | ||
+ | * // | ||
+ | * // | ||
+ | 这两种需求实际上体现了在**运行期或者编译期**绑定 deleter 的区别。 | ||
+ | ==在运行期绑定 deleter== | ||
+ | 由于 // | ||
+ | <code cpp> | ||
+ | // value of del known only at run time; call through a pointer | ||
+ | del ? del(p) : delete p; // del(p) requires run-time jump to del's location | ||
+ | </ | ||
+ | 只要指向 deleter 的指针不为空,就调用 deleter 进行析构操作。 | ||
+ | ==在编译期绑定 deleter== | ||
+ | deleter 是 // | ||
+ | <code cpp> | ||
+ | // del bound at compile time; direct call to the deleter is instantiated | ||
+ | del(p); | ||
+ | </ | ||
+ | ====模板参数推断==== | ||
+ | **模板的参数推断**指编译器会根据调用中的 arugment 的类型来确定其模板参数的类型,并用该模板参数来生成满足调用的函数实例。 | ||
+ | ===类型类模板参数与类型转换=== | ||
+ | 当函数的 parameter 是模板参数时: | ||
+ | * 被传递的 argument 如果是 top-const,依然会被忽略。 | ||
+ | * 只有**两种**类型转换会被执行: | ||
+ | *** non-const** 的对象传递(绑定)到 low-const 的 parameter(reference & pointer to const) | ||
+ | *在函数 parameter 类型**不是引用**(非引用传递)的前提下,传递函数或者数组参数将转化为**指针**进行传递。引用传递下,数组将以数组类型被传递(包括数组的大小) | ||
+ | <code cpp> | ||
+ | template < | ||
+ | template < | ||
+ | |||
+ | string s1("a value" | ||
+ | const string s2(" | ||
+ | |||
+ | fobj(s1, s2); // calls fobj(string, | ||
+ | fref(s1, s2); // calls fref(const string&, | ||
+ | // as a non-const object, s1 will be bind to an reference to const string parameter | ||
+ | | ||
+ | int a[10], b[42]; | ||
+ | fobj(a, b); // calls f(int*, int*) | ||
+ | // error: array are not converted to pointer here, because fref takes reference type parameter. | ||
+ | // as a result, array are passed as array type. since size of array is a part of its type, thus types don't match here. | ||
+ | fref(a, b); | ||
+ | </ | ||
+ | ==函数参数使用相同的模板参数类型的情况== | ||
+ | 对于相同的模板参数类型,需要确保最终推断的结果类型完全一致,否则会导致调用错误。比如: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | bool compare(T, T) {....} | ||
+ | long lng; | ||
+ | compare(lng, | ||
+ | </ | ||
+ | 由于模板参数的类型转换中不存在算术类型的转换,因此模板推断结果冲突,调用失败。这种情况下可以使用不同的模板参数来解决: | ||
+ | <code cpp> | ||
+ | // argument types can differ but must be compatible | ||
+ | template < | ||
+ | int flexibleCompare(const A& v1, const B& v2) | ||
+ | { | ||
+ | if (v1 < v2) return -1; | ||
+ | if (v2 < v1) return 1; | ||
+ | return 0; | ||
+ | } | ||
+ | </ | ||
+ | 上述的 '' | ||
+ | ==混用模板参数与普通参数的情况== | ||
+ | 函数的 parameter 可以混用模板参数与普通参数。此时普通参数的类型转换遵循普通参数的规则(第六章中的 //normal conversion// | ||
+ | ===显式指定模板参数类型=== | ||
+ | 两种情况下需要用户显式的指定模板参数的类型: | ||
+ | * 模板参数类型作无法被编译器推断,比如作为返回值的时候 | ||
+ | * 模板参数类型需要适应用户的需求,用于创建用户指定类型的实例 | ||
+ | ==如何显式指定模板实参类型== | ||
+ | 带模板的函数(类)使用时,函数名后的**三角括号**中就是显式指定模板参数类型的地方: | ||
+ | <code cpp> | ||
+ | // T1 cannot be deduced: it doesn' | ||
+ | template < | ||
+ | T1 sum(T2, T3); | ||
+ | |||
+ | // T1 is explicitly specified; T2 and T3 are inferred from the argument types | ||
+ | auto val3 = sum<long long>(i, lng); // long long sum(int, long) | ||
+ | </ | ||
+ | ==多个需要指定类型的参数的情况== | ||
+ | * 三角括号中指定的参数类型会**从左到右**依次匹配模板参数。 | ||
+ | * 如果指定的参数类型数量不够,则右边的模板参数不会被指定类型 | ||
+ | 需要注意的是,如果调整了函数中模板参数的**位置**,那么**所有的参数类型必须被指定**: | ||
+ | <code cpp> | ||
+ | // poor design: users must explicitly specify all three template parameters | ||
+ | template < | ||
+ | T3 alternative_sum(T2, | ||
+ | |||
+ | // error: can't infer initial template parameters | ||
+ | auto val3 = alternative_sum< | ||
+ | // ok: all three parameters are explicitly specified | ||
+ | auto val2 = alternative_sum< | ||
+ | </ | ||
+ | ==显式指定的实参遵循普通的类型转换== | ||
+ | 当模板参数类型被显式指定时,该参数对应的 argument 遵循普通的类型转换。这一点非常实用,可以允许我们突破参数类型推断下的类型冲突: | ||
+ | <code cpp> | ||
+ | long lng; | ||
+ | compare(lng, | ||
+ | compare< | ||
+ | compare< | ||
+ | </ | ||
+ | ===尾置返回类型与转换模板类型=== | ||
+ | 显式指定模板类型的前提是**用户知道**应该指定什么类型。但某些情况下,模板类型无法被推断,也无法由用户判断。这种情况下,解决的方式有两种: | ||
+ | * 以引用的形式返回返回值 | ||
+ | * 使用尾置返回类型(C++ )判断返回值类型 | ||
+ | ==以引用形式返回未知类型== | ||
+ | 通常情形中,造成无法判断类型的主要原因是使用**间接方式**访问了对象。这种情况下可以以引用的方式返回对象,我们只需要知道这是一种绑定返回对象的引用即可: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | ??? &fcn(It beg, It end) | ||
+ | { | ||
+ | // process the range | ||
+ | return *beg; // return a reference to an element from the range | ||
+ | } | ||
+ | |||
+ | vector< | ||
+ | Blob< | ||
+ | auto &i = fcn(vi.begin(), | ||
+ | auto &s = fcn(ca.begin(), | ||
+ | </ | ||
+ | ==以尾置返回形式判断返回类型== | ||
+ | C++11 中提供的尾置声明可以用于这种情形,即通过 '' | ||
+ | <code cpp> | ||
+ | // a trailing return lets us declare the return type after the parameter list is seen | ||
+ | template < | ||
+ | auto fcn(It beg, It end) -> decltype(*beg) //decltype return a reference on lvalue | ||
+ | { | ||
+ | // process the range | ||
+ | return *beg; // return a reference to an element from the range | ||
+ | } | ||
+ | </ | ||
+ | ==Type Transformation== | ||
+ | 尾置返回的方法还存在一个缺陷。由于 '' | ||
+ | \\ \\ \\ {{ : | ||
+ | 此处我们使用 '' | ||
+ | <code cpp> | ||
+ | // must use typename to use a type member of a template parameter; see § 16.1.3 (p. 670) | ||
+ | template < | ||
+ | auto fcn2(It beg, It end) -> | ||
+ | typename remove_reference< | ||
+ | { | ||
+ | // process the range | ||
+ | return *beg; // return a copy of an element from the range | ||
+ | } | ||
+ | </ | ||
+ | 这里的 '' | ||
+ | ===函数指针与实参推断=== | ||
+ | 使用函数模板对函数指针进行初始化或者赋值时,编译器会利用指针的类型来推断模板的 argument: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | // pf1 points to the instantiation int compare(const int&, const int&) | ||
+ | int (*pf1)(const int&, const int&) = compare; | ||
+ | </ | ||
+ | '' | ||
+ | <code cpp> | ||
+ | // overloaded versions of func; each takes a different function pointer type | ||
+ | void func(int(*)(const string&, | ||
+ | void func(int(*)(const int&, const int&)); | ||
+ | func(compare); | ||
+ | </ | ||
+ | 上例中,函数 '' | ||
+ | <code cpp> | ||
+ | // ok: explicitly specify which version of compare to instantiate | ||
+ | func(compare< | ||
+ | </ | ||
+ | <WRAP center round info 100%> | ||
+ | 模板的实例化开始时,**每个**模板参数用于实例化的类型(或者是值)**都必须唯一**。 | ||
+ | </ | ||
+ | ===模板实参的推断和引用=== | ||
+ | 当模板参数对应的函数参数是引用的时候,有两种情况需要讨论: | ||
+ | * 函数参数为左值引用 | ||
+ | * 函数参数为右值引用 | ||
+ | ==函数参数为 T& 的推断== | ||
+ | 当函数参数类型为 '' | ||
+ | * 函数**只能接受左值**(比如以引用形式返回的值) | ||
+ | * 推断结果为 argument 本身的类型 | ||
+ | * 如果 argument 带 const,那么推断出来的类型也带 const | ||
+ | <code cpp> | ||
+ | template < | ||
+ | // calls to f1 use the referred-to type of the argument as the template parameter type | ||
+ | f1(i); | ||
+ | f1(ci); | ||
+ | f1(5); | ||
+ | </ | ||
+ | ==函数参数类型为 const T& 的推断== | ||
+ | 当函数参数类型为 '' | ||
+ | * 函数可以接收**任意类型**的 argument | ||
+ | * 推断结果为 argument 本身的类型(不附带 constness) | ||
+ | * 由于 const 是函数参数类型的一部分,因此无论 argument 是否为 const, '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | // parameter in f2 is const &; const in the argument is irrelevant | ||
+ | // in each of these three calls, f2's function parameter is inferred as const int& | ||
+ | f2(i); | ||
+ | f2(ci); // ci is a const int, but template parameter T is int | ||
+ | f2(5); | ||
+ | </ | ||
+ | |||
+ | ==函数参数类型为T&& | ||
+ | 当函数参数类型为 '' | ||
+ | <WRAP center round important 100%> | ||
+ | 函数参数类型**为右值引用的模板类型**,在接受 argument 时,与普通右值引用参数最大的不同,是其可以**接收左值**。 | ||
+ | </ | ||
+ | * 函数可以接收**右值**,**以及左值** | ||
+ | * 接收类型为**右值**的 argument 时,推断类型是右值本身的类型 '' | ||
+ | * 接受类型为**普通左值**的 argument 时,推断类型是左值引用 '' | ||
+ | * 接收类型为**常量左值**的 argument 时,推断类型是**常量左值引用** '' | ||
+ | \\ {{ : | ||
+ | ==T&& | ||
+ | 模板右值引用类型的函数可以接收左值的原因是因为**引用折叠**(// | ||
+ | - 当类型为 '' | ||
+ | - 此时,编译器通过**间接的方式**(// | ||
+ | - C++ 11 中规定,如果引用的绑定中出现了**间接的**引用到引用的绑定,那么这种绑定会导致推断出的类型被**折叠为左值引用** '' | ||
+ | 上述第三部也就是是我们之前提到的**引用折叠**。实际上,引用折叠会发生在所有的**引用以间接的方式绑定引用**的场景中: | ||
+ | * 左值引用绑定左值引用 '' | ||
+ | * 右值引用绑定左值引用 '' | ||
+ | * 左值引用绑定右值引用 '' | ||
+ | * 右值引用绑定右值引用 '' | ||
+ | 折叠后的类型推断结果可以总结为: | ||
+ | * 只要引用相互绑定中**有左值引用**,最后的推断类型**都是左值引用** '' | ||
+ | * 如果是 '' | ||
+ | 来看一下 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | f3(42); // argument is an rvalue of type int; template parameter T is int | ||
+ | |||
+ | f3(i); | ||
+ | f3(ci); // argument is an lvalue; template parameter T is const int& | ||
+ | </ | ||
+ | 当参数类型是左值 '' | ||
+ | <code cpp> | ||
+ | // invalid code, for illustration purposes only | ||
+ | void f3< | ||
+ | </ | ||
+ | 但最终被折叠为 '' | ||
+ | <code cpp> | ||
+ | void f3< | ||
+ | </ | ||
+ | <WRAP center round box 100%> | ||
+ | 模板类型右值引用参数可以接受左值意味着该类型的参数实际上可以接收任意类型的 argument;也就是说这种情况下,可以被当成右值使用的 argument,也可以被当做左值使用。 | ||
+ | </ | ||
+ | ==实例分析:函数模板与右值引用参数== | ||
+ | 以下模板函数中: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | { | ||
+ | T t = val; // copy or binding a reference? | ||
+ | t = fcn(t); // does the assignment change only t or val and t? | ||
+ | if (val == t) { /* ... */ } // always true if T is a reference type | ||
+ | } | ||
+ | </ | ||
+ | * 当 '' | ||
+ | * 当 '' | ||
+ | <WRAP center round box 100%> | ||
+ | 实践中,右值引用参数主要用于两种情况: | ||
+ | * 模板**转发**其 argument | ||
+ | * 模板**重载**(与构造函数的重载类似) | ||
+ | </ | ||
+ | ===理解 std:: | ||
+ | '' | ||
+ | <code cpp> | ||
+ | // for the use of typename in the return type and the cast see § 16.1.3 (p. 670) | ||
+ | // remove_reference is covered in § 16.2.3 (p. 684) | ||
+ | template < | ||
+ | typename remove_reference< | ||
+ | { | ||
+ | // static_cast covered in § 4.11.3 (p. 163) | ||
+ | | ||
+ | } | ||
+ | </ | ||
+ | 考虑到 '' | ||
+ | * 该函数应允许任意类型的 argument | ||
+ | * 该函数应该最终将输入的 argument 类型转化为其对应类型的右值引用 | ||
+ | 可以看到的是,'' | ||
+ | 那么,上面的设计的思路就很明显了: | ||
+ | - 通过右值引用的函数模板参数,允许 '' | ||
+ | - 通过 ’‘remove_reference’‘ 的实例化去除 argument 可能存在的引用 | ||
+ | - 通过 '' | ||
+ | |||
+ | ==std::move 是如何运作的== | ||
+ | 考虑下面代码: | ||
+ | <code cpp> | ||
+ | string s1(" | ||
+ | s2 = std:: | ||
+ | s2 = std:: | ||
+ | </ | ||
+ | 当 '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * 因此 '' | ||
+ | * 返回的是 '' | ||
+ | 当 '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * 因此 '' | ||
+ | * 最终返回执行 '' | ||
+ | * 由于 '' | ||
+ | <WRAP center round box 100%> | ||
+ | static_cast 可以转换左值到右值引用又一次体现了 C++ 的设计哲学;提供灵活性,但前提是你要知道你自己在做什么。比如这类情况,转换的前提是要求左值可以被安全的截断;而 static_cast 又是一种显式的强制转换。C++ 通过这种方式将处理权完全移交给了程序员,但同时又通过显式的方式来提醒程序员。 | ||
+ | </ | ||
+ | ===转发 / Forwarding=== | ||
+ | 在实际应用中,某些函数需要将接收的 argument 原封不动的转交给其他函数。具体的说,我们希望保存以下的特性: | ||
+ | * 左值 / 右值 | ||
+ | * constness | ||
+ | ==实例:argument 转发过程中丢失信息== | ||
+ | 下面是一个转发过程中 argument 发生变化的例子: | ||
+ | <code cpp> | ||
+ | // template that takes a callable and two parameters | ||
+ | // and calls the given callable with the parameters '' | ||
+ | // flip1 is an incomplete implementation: | ||
+ | template < | ||
+ | void flip1(F f, T1 t1, T2 t2) | ||
+ | { | ||
+ | f(t2, t1); | ||
+ | } | ||
+ | |||
+ | void f(int v1, int &v2) // note v2 is a reference | ||
+ | { | ||
+ | cout << v1 << " " << ++v2 << endl; | ||
+ | } | ||
+ | |||
+ | f(42, i); // f changes its argument i | ||
+ | flip1(f, j, 42); // f called through flip1 leaves j unchanged | ||
+ | </ | ||
+ | 上面是一个函数 '' | ||
+ | \\ \\ \\ {{ : | ||
+ | * 由于 '' | ||
+ | * 那么 '' | ||
+ | <code cpp> | ||
+ | void flip1(void(*fcn)(int, | ||
+ | </ | ||
+ | 因此传递给 '' | ||
+ | ==改良:使用右值引用== | ||
+ | 上面的例子中,存在(潜在)的问题有: | ||
+ | * 左值在拷贝的过程中成了右值('' | ||
+ | * 以拷贝的方式传递参数也可能导致 constness 的变化 | ||
+ | 根据右值引用以下的特性,将 '' | ||
+ | * 右值引用接受左值,并以引用的形式传递左值 | ||
+ | * 右值引用接受右值,会以右值引用的形式传递右值 | ||
+ | * 传递过程中,constness 的信息不会丢失 | ||
+ | 比如下面的 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | void flip2(F f, T1 && | ||
+ | { | ||
+ | f(t2, t1); | ||
+ | } | ||
+ | </ | ||
+ | 在之前的调用中: | ||
+ | \\ \\ < | ||
+ | '' | ||
+ | ==潜在的问题== | ||
+ | 上述的方法中存在一个问题: | ||
+ | * '' | ||
+ | * 具体函数的类型转换中**不存在引用折叠** | ||
+ | 这带来的问题是,如果普通函数中的参数类型是右值引用,而上游传递的参数类型是左值(引用),那么**普通函数是无法完成左值到右值的转换**的。比如将 '' | ||
+ | <code cpp> | ||
+ | void g(int && | ||
+ | { | ||
+ | cout << i << " " << j << endl; | ||
+ | } | ||
+ | </ | ||
+ | 这种情况下,我们无法将一个左值(引用) '' | ||
+ | \\ \\ \\ < | ||
+ | ==使用 std:: | ||
+ | 标准哭中提供了模板 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | { | ||
+ | finalFcn(std:: | ||
+ | // ... | ||
+ | } | ||
+ | </ | ||
+ | 其实现原理也使用了引用折叠: | ||
+ | * 当接受的参数是左值时,发生了两次引用折叠: | ||
+ | * 第一次引用折叠推断模板类型为 '' | ||
+ | * 第二次引用折叠发生在返回:返回的类型是 '' | ||
+ | * 当接受参数是右值: | ||
+ | * 模板参数被推断为 '' | ||
+ | * 返回的类型是 '' | ||
+ | \\ | ||
+ | {{ : | ||
+ | \\ \\ | ||
+ | <WRAP center round box 100%> | ||
+ | '' | ||
+ | * 利用模板参数的引用折叠引用性质,使用了模板来**避免普通右值引用无法接收左值**的问题 | ||
+ | * 同时也通过利用引用折叠,保证了传递进来的 argument 的**左右值特性不会发生变化** | ||
+ | </ | ||
+ | <WRAP center round info 100%> | ||
+ | // | ||
+ | </ | ||
+ | ====函数模板与重载==== | ||
+ | 当函数模板参与到重载函数的匹配中时: | ||
+ | * 所有**成功实例化**的模板**实例**均是 //candidate function// | ||
+ | * 属于 //candidate function// 的实例同时也是 //viable function// | ||
+ | * 关于 //viable function// 的匹配等级标准也适用于函数模板 | ||
+ | 当同时存在几个最佳匹配结果的时候,还需要额外考虑: | ||
+ | - 候选者中,如果只有一个**非模板函数**,那么该函数会被选择 | ||
+ | - 如果没有非模板函数,则选择更加**特殊**的版本(处理范围更窄的版本) | ||
+ | - 否则,调用会具有二义性 | ||
+ | ===实例:模板重载与匹配规则=== | ||
+ | 以书上的内容为例,实现一个 '' | ||
+ | ==只存在唯一 viable 的情况== | ||
+ | 首先,我们实现两个版本的 '' | ||
+ | * 第一个的参数类型是 '' | ||
+ | * 第二个的参数类型是 '' | ||
+ | 实现如下: | ||
+ | <code cpp> | ||
+ | // print any type we don't otherwise handle | ||
+ | template < | ||
+ | { | ||
+ | ostringstream ret; // see § 8.3 (p. 321) | ||
+ | ret << t; // uses T's output operator to print a representation of t | ||
+ | return ret.str(); // return a copy of the string to which ret is bound | ||
+ | } | ||
+ | |||
+ | // print pointers as their pointer value, followed by the object to which the pointer points | ||
+ | template < | ||
+ | { | ||
+ | ostringstream ret; | ||
+ | ret << " | ||
+ | if (p) | ||
+ | ret << " " << debug_rep(*p); | ||
+ | else | ||
+ | ret << " null pointer"; | ||
+ | return ret.str(); // return a copy of the string to which ret is bound | ||
+ | } | ||
+ | </ | ||
+ | 如果进行以下调用: | ||
+ | <code cpp> | ||
+ | string s(" | ||
+ | cout << debug_rep(s) << endl; | ||
+ | </ | ||
+ | '' | ||
+ | * 第一个 '' | ||
+ | * 第二个 '' | ||
+ | 因此,此处选则 '' | ||
+ | ==存在多个 viable 的情况== | ||
+ | 上述实现中,如果将调用改变为: | ||
+ | <code cpp> | ||
+ | cout << debug_rep(& | ||
+ | </ | ||
+ | 那么两个版本的函数模板都会成为 // | ||
+ | * '' | ||
+ | * '' | ||
+ | 由于 '' | ||
+ | ==存在特殊版本的情况== | ||
+ | 将上面的调用改为: | ||
+ | <code cpp> | ||
+ | const string *sp = &s; | ||
+ | cout << debug_rep(sp) << endl; | ||
+ | </ | ||
+ | * '' | ||
+ | * '' | ||
+ | 由于编译器无法分辨指针与指针的引用之间的区别,此时两者均是普通意义上的最佳匹配(exactly match,普通函数到这一步已经会导致二义性了)。但需要注意的是,由于 '' | ||
+ | ==存在唯一非模板函数 viable 的情况== | ||
+ | 我们再为 '' | ||
+ | <code cpp> | ||
+ | // print strings inside double quotes | ||
+ | string debug_rep(const string &s) | ||
+ | { | ||
+ | return '"' | ||
+ | } | ||
+ | </ | ||
+ | 当以下面命令进行调用时: | ||
+ | <code cpp> | ||
+ | string s(" | ||
+ | cout << debug_rep(s) << endl; | ||
+ | </ | ||
+ | 候选有两个: | ||
+ | * '' | ||
+ | * '' | ||
+ | 此处提供的参数类型为 '' | ||
+ | ==复合情况== | ||
+ | 考虑以下调用: | ||
+ | <code cpp> | ||
+ | cout << debug_rep(" | ||
+ | </ | ||
+ | * '' | ||
+ | * '' | ||
+ | * 普通 '' | ||
+ | 由于普通非模板版本中存在低等级的类型转换,因此首先排除该版本。到此,'' | ||
+ | 实际实现中,我们更倾向于为 '' | ||
+ | <code cpp> | ||
+ | // convert the character pointers to string and call the string version of debug_rep | ||
+ | string debug_rep(char *p) | ||
+ | { | ||
+ | return debug_rep(string(p)); | ||
+ | } | ||
+ | string debug_rep(const char *p) | ||
+ | { | ||
+ | return debug_rep(string(p)); | ||
+ | } | ||
+ | </ | ||
+ | ==有关函数前置声明是非常必要的== | ||
+ | 上面的例子中,我们使用 '' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | template < | ||
+ | // the following declaration must be in scope | ||
+ | // for the definition of debug_rep(char*) to do the right thing | ||
+ | string debug_rep(const string &); | ||
+ | string debug_rep(char *p) | ||
+ | { | ||
+ | // if the declaration for the version that takes a const string& is not in scope | ||
+ | // the return will call debug_rep(const T&) with T instantiated to string | ||
+ | return debug_rep(string(p)); | ||
+ | } | ||
+ | </ | ||
+ | 这是因为对于未声明函数的调用,普通函数与成员函数的处理方式不同: | ||
+ | * 在普通函数中,使用未声明的函数是语法错误行为 | ||
+ | * 在函数模板中,如果有任意重载版本可以使该函数实例化,那么编译器不会报错。 | ||
+ | 上面的例子中,如果我们不对普通的 '' | ||
+ | ====Variadic 模板==== | ||
+ | C++11 提供了 //Variadic template// | ||
+ | * 模板参数包(// | ||
+ | * 函数参数包(// | ||
+ | //Variadic template// 通过省略号参数(// | ||
+ | <code cpp> | ||
+ | // Args is a template parameter pack; rest is a function parameter pack | ||
+ | // Args represents zero or more template type parameters | ||
+ | // rest represents zero or more function parameters | ||
+ | template < | ||
+ | void foo(const T &t, const Args& ... rest); | ||
+ | </ | ||
+ | 在对 //Variadic template// 进行类型推断的时候,编译器还会根据 argument 的**数量**来推断参数包中参数的数量: | ||
+ | <code cpp> | ||
+ | int i = 0; double d = 3.14; string s = "how now brown cow"; | ||
+ | foo(i, s, 42, d); // three parameters in the pack | ||
+ | foo(s, 42, " | ||
+ | foo(d, s); // one parameter in the pack | ||
+ | foo(" | ||
+ | |||
+ | /* instantiation */ | ||
+ | void foo(const int&, const string&, | ||
+ | void foo(const string&, | ||
+ | void foo(const double&, | ||
+ | void foo(const char(& | ||
+ | </ | ||
+ | ==计算包中的参数数量== | ||
+ | 参数包中的参数数量可以通过 '' | ||
+ | <code cpp> | ||
+ | template< | ||
+ | cout << sizeof...(Args) << endl; // number of type parameters | ||
+ | cout << sizeof...(args) << endl; // number of function parameters | ||
+ | } | ||
+ | </ | ||
+ | <WRAP center round box 100%> | ||
+ | '' | ||
+ | </ | ||
+ | |||
+ | ===实例:打印不同数量类型的参数=== | ||
+ | C++ 可以通过 '' | ||
+ | \\ \\ | ||
+ | 在设计上,由于 //Variadic template// 经常与递归函数搭配。这是因为参数包的处理分为两步: | ||
+ | * 当前层级处理 pack 中的第一个元素 | ||
+ | * pack 中剩余的参数通过递归的方式送到下一个层级展开 | ||
+ | 因此,我们的打印函数 '' | ||
+ | <code cpp> | ||
+ | // function to end the recursion and print the last element | ||
+ | // this function must be declared before the variadic version of print is defined | ||
+ | template< | ||
+ | ostream & | ||
+ | { | ||
+ | return os << t; // no separator after the last element in the pack | ||
+ | } | ||
+ | // this version of print will be called for all but the last element in the pack | ||
+ | template < | ||
+ | ostream & | ||
+ | { | ||
+ | os << t << ", "; | ||
+ | return print(os, rest...); // recursive call; print the other arguments | ||
+ | } | ||
+ | </ | ||
+ | ==variadic 版本的 print 的设计== | ||
+ | 上面的实现中,variadic 版本的 '' | ||
+ | <code cpp> | ||
+ | ostream & | ||
+ | </ | ||
+ | 而其内部的递归调用中,'' | ||
+ | <code cpp> | ||
+ | return print(os, rest...); | ||
+ | </ | ||
+ | 之所以可以这么做,当 argument 以参数包的形式进行传递时,可以以参数组合的方式进行接收。也就是说,我们传递到下个层级 ’‘print()’‘ 参数包 '' | ||
+ | * 参数包中的第一个参数(在当前的层级中打印出来) | ||
+ | * 参数包中剩余的参数 | ||
+ | \\ < | ||
+ | 通这样的递归调用,参数包中的参数将逐步减少。当参数包中只存在一个参数时,候选这将会有两个。比如下面的调用: | ||
+ | <code cpp> | ||
+ | print(cout, i, s, 42); // two parameters in the pack | ||
+ | </ | ||
+ | 该递归最后一次调用将只有一个参数: | ||
+ | <code cpp> | ||
+ | print(cout, 42); | ||
+ | </ | ||
+ | 此时,我们的 variadic 版本 和 non-variadic 版本的 '' | ||
+ | <WRAP center round box 100%> | ||
+ | 习题中有一个问题,如果 non-variadic 版本 对 variadic 版本不可见会导致什么后果?\\ | ||
+ | 书上给出了 warning,说会导致无限递归:\\ | ||
+ | >// A declaration for the nonvariadic version of print must be in scope when the variadic version is defined. Otherwise, the variadic function will recurse indefinitely.// | ||
+ | 但在实践过程中,编译无法通过,会报没有匹配函数的错误。这一点其实不难解释:从之前的例子可以看出来,当参数包为空的时候并不会 “占用” 参数的位置。如果不存在 non-variadic 的版本,那么该递归将继续进行到 '' | ||
+ | </ | ||
+ | ===参数包的扩展=== | ||
+ | 我们可以通过提供指定的 pattern 对参数包进行扩展: | ||
+ | * 扩展定义方式:'' | ||
+ | * 扩展结果:pattern 将会应用到参数包内每个元素中 | ||
+ | <code cpp> | ||
+ | template < | ||
+ | ostream & | ||
+ | print(ostream &os, const T &t, const Args& | ||
+ | { | ||
+ | os << t << ", "; | ||
+ | return print(os, rest...); | ||
+ | } | ||
+ | </ | ||
+ | 上例中,我们对 '' | ||
+ | * 在函数参数中:模板参数列表提供的类型参数包,被扩展为了带引用的类型参数包(Args-> | ||
+ | * 在调用中:rest 本身作为 pattern,扩展的效果是生成以逗号分隔的元素列表: | ||
+ | 而通过扩展,'' | ||
+ | <code cpp> | ||
+ | print(cout, i, s, 42); // two parameters in the pack | ||
+ | // | ||
+ | ostream& | ||
+ | // | ||
+ | print(os, s, 42); | ||
+ | </ | ||
+ | ==以函数作为 pattern== | ||
+ | 函数也可以作为 pattern 应用到参数包中的每一个元素上: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | ostream & | ||
+ | { | ||
+ | // print(os, debug_rep(a1), | ||
+ | return print(os, debug_rep(rest)...); | ||
+ | } | ||
+ | </ | ||
+ | 上例 '' | ||
+ | <code cpp> | ||
+ | errorMsg(cerr, | ||
+ | </ | ||
+ | 实际上是调用了: | ||
+ | <code cpp> | ||
+ | print(cerr, debug_rep(a), | ||
+ | </ | ||
+ | <WRAP center round info 100%> | ||
+ | 注意区分两种写法: | ||
+ | * // | ||
+ | * // | ||
+ | </ | ||
+ | ===转发与参数包=== | ||
+ | 由于 //variadic template// 往往是基于已有的工具函数进行适配,参数传递在其实现过程中是很常见的。为了保证参数传递的准确性,// | ||
+ | ==emplace_back() 的实现== | ||
+ | // | ||
+ | <code cpp> | ||
+ | /* declaration */ | ||
+ | class StrVec { | ||
+ | public: | ||
+ | template < | ||
+ | // remaining members as in § 13.5 (p. 526) | ||
+ | }; | ||
+ | /* defination */ | ||
+ | template < | ||
+ | inline | ||
+ | void StrVec:: | ||
+ | { | ||
+ | chk_n_alloc(); | ||
+ | alloc.construct(first_free++, | ||
+ | } | ||
+ | </ | ||
+ | 上例中的两个个要点: | ||
+ | - 在函数的声明中对函数参数的类型扩展为了**右值引用** | ||
+ | - 在 '' | ||
+ | 其中第二个要点: | ||
+ | <code cpp> | ||
+ | std:: | ||
+ | </ | ||
+ | 实际上指对于每个实例化的类型,以及其对应的参数,都进行转发: | ||
+ | <code cpp> | ||
+ | //Ti represents the type of the ith element in the template parameter pack | ||
+ | //ti represents the ith element in the function parameter pack | ||
+ | std:: | ||
+ | </ | ||
+ | 比如: | ||
+ | <code cpp> | ||
+ | svec.emplace_back(10, | ||
+ | </ | ||
+ | 实际上参数包被拓展为: | ||
+ | <code cpp> | ||
+ | std:: | ||
+ | </ | ||
+ | 也就是以当前地址为起始,构建 10 个内容为 '' | ||
+ | <WRAP center round box 100%> | ||
+ | //Variadic Templates// + // | ||
+ | * 参数包是被扩展的右值引用包 | ||
+ | * 参数包的传递会经过 '' | ||
+ | 其优势在于,可以接受任何数量/ | ||
+ | </ | ||
+ | ====模板的特化==== | ||
+ | 单个模板往往不能处理所有的问题。当某些类型需要特殊处理时,我们可以基于当前的模板,为其**定义**一个额外的模板版本进行处理。定义这个模板的过程被成为模板的**特化**(// | ||
+ | ===特化模板的定义=== | ||
+ | 以下面的例子为例: | ||
+ | <code cpp> | ||
+ | template < | ||
+ | </ | ||
+ | 通常情况下,用于比较的两个对象都是通过运算符来实现的。但当我们需要比较两个 literal string 时,用于比较的方法是 '' | ||
+ | <code cpp> | ||
+ | // special version of compare to handle pointers to character arrays | ||
+ | template <> | ||
+ | int compare(const char* const &p1, const char* const &p2) | ||
+ | { | ||
+ | return strcmp(p1, p2); | ||
+ | } | ||
+ | </ | ||
+ | 特化版本以 '' | ||
+ | * 特化版本需要处理的是 '' | ||
+ | * 标准版本中是 '' | ||
+ | * 此处需要将 '' | ||
+ | * 因此最终类型是 '' | ||
+ | ==函数的重载与特化== | ||
+ | * 特化是对模板的**指定实例化**,是用户代替编译器实现实例化的过程 | ||
+ | * **特化不是重载**,因此不会参与重载函数的匹配 | ||
+ | <WRAP center round box 100%> | ||
+ | 需要特别注意的是,特化遵循普通的作用域规则。有几点要注意: | ||
+ | * 特化版本的源模板声明要对特化版本的声明可见 | ||
+ | * 特化版本的声明要对其使用可见。否则,编译器将会使用源模板进行实例化,这种错误非常难查。 | ||
+ | * 特化版本不能使用与源模板版本相同的参数。编译器无法查出此类错误 | ||
+ | </ | ||
+ | <WRAP center round tip 100%> | ||
+ | 推荐的做法: | ||
+ | * 将特化版本与其源模板声明在同一个头文件中 | ||
+ | * 所有的源模板的声明需要置于所有的特化版本之前 | ||
+ | </ | ||
+ | ===类模板的特化=== | ||
+ | 类模板也可以针对特定的类型进行特化。 | ||
+ | ==实例:使用类模板特化生成 Sales_data 类型的哈希函数== | ||
+ | 在[[cs: | ||
+ | 特化标准库的哈希函数有几个准备工作: | ||
+ | * 需要重载调用运算符 '' | ||
+ | * alias 两个类型:'' | ||
+ | * 默认构造函数与拷贝赋值运算符(可使用隐式生成版本) | ||
+ | 除此之外,由于 '' | ||
+ | <code cpp> | ||
+ | // open the std namespace so we can specialize std::hash | ||
+ | namespace std { | ||
+ | } // close the std namespace; note: no semicolon after the close curly | ||
+ | </ | ||
+ | 下面是实现方式: | ||
+ | <code cpp> | ||
+ | // open the std namespace so we can specialize std::hash | ||
+ | namespace std { | ||
+ | template <> | ||
+ | struct hash< | ||
+ | { | ||
+ | // the type used to hash an unordered container must define these types | ||
+ | typedef size_t result_type; | ||
+ | typedef Sales_data argument_type; | ||
+ | size_t operator()(const Sales_data& | ||
+ | // our class uses synthesized copy control and default constructor | ||
+ | }; | ||
+ | size_t | ||
+ | hash< | ||
+ | { | ||
+ | return hash< | ||
+ | | ||
+ | | ||
+ | } | ||
+ | } // close the std namespace; note: no semicolon after the close curly | ||
+ | </ | ||
+ | 几个要点: | ||
+ | * '' | ||
+ | * 特化版本对源版本的匹配工作: | ||
+ | * 需要为源模板的构造函数参数 '' | ||
+ | * 返回值与源版本需要一致 | ||
+ | * '' | ||
+ | * 该特化需要与 '' | ||
+ | <WRAP center round box 100%> | ||
+ | 特化后的类 / 函数不再是模板。本例中,因为特化版本定义在了 '' | ||
+ | </ | ||
+ | ===类模板的偏特化=== | ||
+ | 类模板的特化允许只为源模板提供一部分 argument;这种方式的特化被成为**类模板的偏特化**(// | ||
+ | * 必要的 argument 必须提供 | ||
+ | * 只有类模板可以进行片特化 | ||
+ | ==实例以及应用== | ||
+ | Type transformation 模板中的 '' | ||
+ | <code cpp> | ||
+ | // original, most general template | ||
+ | template <class T> struct remove_reference { | ||
+ | typedef T type; | ||
+ | }; | ||
+ | // partial specializations that will be used for lvalue and rvalue references | ||
+ | template <class T> struct remove_reference< | ||
+ | { typedef T type; }; | ||
+ | template <class T> struct remove_reference< | ||
+ | { typedef T type; }; | ||
+ | </ | ||
+ | 可以看出来的是: | ||
+ | * 与特化(全特化)最大的不同是,**偏特化依然是模板** | ||
+ | * 偏特化的版本源版本参数类型不一致 | ||
+ | 在调用中,偏特化的类型会优先调用对应版本: | ||
+ | <code cpp> | ||
+ | int i; | ||
+ | // decltype(42) is int, uses the original template | ||
+ | remove_reference< | ||
+ | // decltype(i) is int&, uses first (T&) partial specialization | ||
+ | remove_reference< | ||
+ | // decltype(std:: | ||
+ | remove_reference< | ||
+ | </ | ||
+ | ==单独特化类成员== | ||
+ | C++ 也允许单独对类模板的常规能源进行特化。比如下面的例子,'' | ||
+ | <code cpp> | ||
+ | template < | ||
+ | Foo(const T &t = T()): mem(t) { } | ||
+ | void Bar() { /* ... */ } | ||
+ | T mem; | ||
+ | // other members of Foo | ||
+ | }; | ||
+ | template<> | ||
+ | void Foo< | ||
+ | { | ||
+ | // do whatever specialized processing that applies to int | ||
+ | } | ||
+ | </ | ||
+ | 该 '' | ||
+ | <code cpp> | ||
+ | Foo< | ||
+ | fs.Bar(); | ||
+ | Foo< | ||
+ | fi.Bar(); | ||
+ | </ |