c++学习笔记之成员模板

简介:
任意类(模板或者是飞模板)可以拥有类模板或函数模板的成员,这种成员成为成员模板,成员模板不能为虚。成员模板的一个例子是标准容器的assign成员,接受两个迭代器的assign版本使用模板形参表示其迭代器形参的类型。另一个例子就是接受两个迭代器的容器构造函数。
1 定义成员模板
模板成员声明看起来像任意模板的声明一样。
template <class Type> class Queue{
    public:
        template <class It>Queue(It beg,It end){
            copy_elems(beg,end);
        }
        template <class Iter> void assign(Iter,Iter);
    private:
        template <class Iter> void copy_elems(Iter,Iter);
};

template <class Type> template <class Iter>
void Queue<Type>::assign(Iter beg,Iter end){
    //destroy();
    copy_elems(beg,end);
}

template <class Type> template <class Iter>
void Queue<Type>::copy_elems(Iter beg,Iter end){
    while(beg != end){
        push(*beg);
        ++beg;
    }
}
成员声明的开头是自己的模板形参表。构造函数和assign函数各有一个模板类型形参,这些函数使用这些类型形参作为其函数形参的类型,他们函数

形参指明要复制元素范围的迭代器。


2 在类外定义成员模板
在类模板作用域外部定义成员模板的时候,那必须包含两个模板形参表:
template <class Type> template <class Iter>
void Queue<Type>::assign(Iter beg,Iter end){
    //destroy();
    copy_elems(beg,end);
}

当成员模板时类模板的成员时,它的定义必须包含类模板形参以及自己的模板形参。
首先是类模板形参表,后面接着是自己模板形参表。
assign函数定义的开头为:
template <class Type> template <class Iter>
第一个模板形参表template <class Type>为类模板的,第二个模板形参表template <class Iter>是成员模板的。
3 成员模板的实例化
成员模板只有在程序使用时才实例化。类模板的成员模板的实例化要比类模板的普通成员函数的实例化要复杂一些。
成员模板有两种模板形参:(1)由类定义的(2)由成员模板本身定义的
类模板形参由调用函数的对象的类型确定,成员定义的模板形参的行为与普通函数模板一样。这些形参都通过常规模版实参推断确定。
short a[4] = {0,3,6,9};
    Queue<int> qi(a,a+4);
    vector<int> vi(a,a+4);
    qi.assign(vi.begin(),vi.end());

因为所构造的是Queue<int>类型的对象,我怕们知道编译器将为Queue<int>实例化基于迭代器构造函数。该构造函数本身模版形参的类型由编译器根据a和a+4的类型推断,而该类型为short指针。因此,qi的定义实例化:
void Queue<int>::Queue(short *,short *);
对assign的调用将实例化qi的成员。qi具有Queue<int>类型,因此,这个调用将实例化名为assign的Queue<int>成员。该函数为函数模版,想起他任意的函数模版一样,编译器从传给调用者的实参推断assign的模版实参,推断得到的类型是vector<int>::iterator,这个调用实例化:
void Queue<int>::assign(vector<int>::iterator,vector<int>::iterator);

目录
相关文章
|
4月前
|
存储 算法 安全
c++模板进阶操作——非类型模板参数、模板的特化以及模板的分离编译
在 C++ 中,仿函数(Functor)是指重载了函数调用运算符()的对象。仿函数可以像普通函数一样被调用,但它们实际上是对象,可以携带状态并具有更多功能。与普通函数相比,仿函数具有更强的灵活性和可扩展性。仿函数通常通过定义一个包含operator()的类来实现。public:// 重载函数调用运算符Add add;// 创建 Add 类的对象// 使用仿函数return 0;
132 0
|
4月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
99 0
|
7月前
|
编译器 C++
模板(C++)
本内容主要讲解了C++中的函数模板与类模板。函数模板是一个与类型无关的函数家族,使用时根据实参类型生成特定版本,其定义可用`typename`或`class`作为关键字。函数模板实例化分为隐式和显式,前者由编译器推导类型,后者手动指定类型。同时,非模板函数优先于同名模板函数调用,且模板函数不支持自动类型转换。类模板则通过在类名后加`&lt;&gt;`指定类型实例化,生成具体类。最后,语录鼓励大家继续努力,技术不断进步!
|
7月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
372 6
|
8月前
|
编译器 C++
㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿
㉿㉿㉿c++模板的初阶(通俗易懂简化版)㉿㉿㉿
|
8月前
|
安全 C++
【c++】模板详解(2)
本文深入探讨了C++模板的高级特性,包括非类型模板参数、模板特化和模板分离编译。通过具体代码示例,详细讲解了非类型参数的应用场景及其限制,函数模板和类模板的特化方式,以及分离编译时可能出现的链接错误及解决方案。最后总结了模板的优点如提高代码复用性和类型安全,以及缺点如增加编译时间和代码复杂度。通过本文的学习,读者可以进一步加深对C++模板的理解并灵活应用于实际编程中。
110 0
|
8月前
|
存储 安全 算法
深入理解C++模板编程:从基础到进阶
在C++编程中,模板是实现泛型编程的关键工具。模板使得代码能够适用于不同的数据类型,极大地提升了代码复用性、灵活性和可维护性。本文将深入探讨模板编程的基础知识,包括函数模板和类模板的定义、使用、以及它们的实例化和匹配规则。
|
11月前
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
342 4
|
8月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
4月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
174 0