从零开始学C++之模板(三):缺省模板参数(借助标准模板容器实现Stack模板)、成员模板、关键字typename

一、缺省模板参数

回顾前面的文章,都是自己管理stack的内存,无论是链栈还是数组栈,能否借助标准模板容器管理呢?答案是肯定的,只需要多传一个模板参数即可,而且模板参数还可以是缺省的,如下:

template <typename T, typename CONT = std::deque<T> > class Stack { …

private:

    CONT c_; };

如果没有传第二个参数,默认为deque 双端队列,当然我们也可以传递std::vector<T>

下面程序借助标准模板容器管理内存来实现stack模板类:

Stack.h:

#ifndef _STACK_H_
#define _STACK_H_

#include <exception>
#include <deque>
using namespace std;

template <typename T, typename CONT = deque<T> >
class Stack
{
public:
    Stack() : c_()
    {
    }
    ~Stack()
    {
    }

    void Push(const T &elem)
    {
        c_.push_back(elem);
    }
    void Pop()
    {
        c_.pop_back();
    }
    T &Top()
    {
        return c_.back();
    }
    const T &Top() const
    {
        return c_.back();
    }
    bool Empty() const
    {
        return c_.empty();
    }
private:
    CONT c_;
};

#endif // _STACK_H_

main.cpp:

#include "Stack.h"
#include <iostream>
#include <vector>
using namespace std;


int main(void)
{
    /*Stack<int> s;*/
    Stack<int, vector<int> > s;
    s.Push(1);
    s.Push(2);
    s.Push(3);

    while (!s.Empty())
    {
        cout << s.Top() << endl;
        s.Pop();
    }
    return 0;
}

输出为 3 2 1

即如果没有传递第二个参数,堆栈和压栈等操作直接调用deque<int> 的成员函数,也由deque<int> 管理内存。

如程序中传递vector<int> ,则由vector<int> 成员函数处理。

二、成员模板

来看下面的例子:

#include <iostream>
using namespace std;


template <typename T>
class MyClass
{
private:
    T value;
public:
    void Assign(const MyClass<T> &x)
    {
        value = x.value;
    }
};

int main(void)
{
    MyClass<double> d;
    MyClass<int> i;

    d.Assign(d);        // OK
    d.Assign(i);        // Error
    return 0;
}

因为i 和 d 的类型不同,故会编译出错。可以用成员模板的方法解决:

#include <iostream>
using namespace std;

template <typename T>
class MyClass
{
private:
    T value;
public:
    MyClass() {}
    template <class X>
    MyClass(const MyClass<X> &x) : value(x.GetValue())
    {

    }
    template <class X>
    void Assign(const MyClass<X> &x)
    {
        value = x.GetValue();
    }
    T GetValue() const
    {
        return value;
    }
};

int main(void)
{
    MyClass<double> d;
    MyClass<int> i;
    d.Assign(d);        // OK
    d.Assign(i);        // OK

    MyClass<double> d2(i);

    return 0;
}

为了支持  MyClass<double> d2(i); 故也要将拷贝构造函数实现为成员模板函数,同理,如果想支持 d = i ; 也要将赋值运算符实现为成员模板。实际上auto_ptr<class> 中的实现就使用了成员模板,因为要支持类似下面的运算:

auto_ptr<X> x;

auto_ptr<Y> y;

x = y;

三、typename 关键字

看下面的例子:

#include <iostream>
using namespace std;

template <typename T>
class MyClass
{
private:
    typename T::SubType *ptr_;
};

class Test
{
public:
    typedef int SubType;
};
int main(void)
{
    MyClass<Test> mc;
    return 0;
}

typename T::SubType *ptr_; 如果前面没有typename 修饰,则SubType会被认为是T类型内部的静态数据成员,推导下去,* 就不再认为是指针,而被

认为是乘号,编译的时候就出错了。加上修饰,就知道SubType 是T 内部的自定义类型,ptr是指向这种类型的指针,编译通过。

在vector 的源码中也可以发现下面的一些片段:

template<> class _CRTIMP2_PURE allocator<void>
{
    // generic allocator for type void
public:
    template<class _Other>
    struct rebind
    {
        // convert an allocator<void> to an allocator <_Other>
        typedef allocator<_Other> other;
    };
    ....
}

typedef typename _Alloc::template rebind<_Ty>::other _Alty;

最后一行是类型定义,由于要解释_Alloc 类型需要引用的代码片段比较多,就姑且认为是allocator<int> 类型,可以看到 rebind<_Ty>

 是成员模板类,other是成员模板类中自定义类型,_Ty 可以认为是int , 那么other 类型也就是allocator<int>, 也就是说_Alty 是类型

allocator<int> 。

此外还可以看到 :

template<class _Ty> class allocator

 { 

template<> class _CRTIMP2_PURE allocator<void>  {  };

};

也就是说allocator<void> 是allocator 模板类的特化。

四、派生类与模板、面向对象与泛型编程

(一)、派生类与模板

1、为了运行的效率,类模板是相互独立的,即独立设计,没有使用继承的思想。对类模板的扩展是采用适配器(adapter)来完成的。通用性是模板库的设计出发点之一,这是由泛型算法(algorithm)和函数对象(functor)等手段达到的。

2、派生的目标之一也是代码的复用和程序的通用性,最典型的就是MFC,派生类的优点是可以由简到繁,逐步深入,程序编制过程中可以充分利用前面的工作,一步步完成一个复杂的任务。

3、模板追求的是运行效率,而派生追求的是编程的效率。

(二)、面向对象与泛型编程

1、面向对象与泛型都依赖于某个形式的多态

面向对象

动态多态(虚函数)

泛型

静态多态(模板类,模板函数)

2、面向对象中的多态在运行时应用存在继承关系。我们编写使用这些类的代码,忽略基类与派生类之间的类型差异。只要使用基类指针或者引用,基类类型对象、派生类类型对象就可以共享相同的代码。

3、在泛型编程中,我们所编写的类和函数能够多态地用于编译时不相关的类型。一个类或一个函数可以用来操纵多种类型的对象。

参考:

C++ primer 第四版 Effective C++ 3rd C++编程规范

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏流媒体

STL算法(for_each/transform)

用于逐个遍历容器元素,它对迭代器区间[first,last)所指的每一个元素,执行由单参数函数对象f所定义的操作。方法返回函数对象。

572
来自专栏coolblog.xyz技术专栏

HashMap 源码详细分析(JDK1.8)

本篇文章我们来聊聊大家日常开发中常用的一个集合类 - HashMap。HashMap 最早出现在 JDK 1.2中,底层基于散列算法实现。HashMap 允许 ...

64924
来自专栏Golang语言社区

Golang语言实现猜数字小游戏的方法

随机生成一个数字,输入一个数字看是否匹对,匹配则结速,反之提示是大了还是小了 package main import ( "bufio" "f...

3419
来自专栏蓝天

C++鲜为人知的符号

这些鲜为人知的C++符号,可直接在代码中使用,但实践中不推荐这么做,可作为茶余饭后的乐趣了解C++的另一面。

822
来自专栏西安-晁州

TypeScript学习笔记之基础类型

从今天开始学习typescript了,记录ts学习点滴,最后,使用ts结合nodejs开发后端应用,一起共勉吧: typescript最新版本2.6,所有演示代...

1990
来自专栏韦弦的微信小程序

Swift 有效的字母异位词 - LeetCode

那么题目就变成了判读两个字符串的组成元素是否是一样的,我们用两个数组获取字符串的所有组成元素,然后排序后比较是否相等:

712
来自专栏Flutter入门

About Kotlin-Kotlin中的类2About Kotlin(2)

使用sealed修饰符修饰。其实是一组类的集合。可以用来表示受限的类的继承结构。 其也可以有子类,所有子类也必须在相同的文件中声明。 密封类从某种意义上说,...

872
来自专栏龙首琴剑庐

Java泛型一览笔录

1、什么是泛型? 泛型(Generics )是把类型参数化,运用于类、接口、方法中,可以通过执行泛型类型调用 分配一个类型,将用分配的具体类型替换泛型类型。然后...

2906
来自专栏乐百川的学习频道

设计模式(十六) 解释器模式

解释器模式是一种行为型模式,它的主要作用是用一种方式来实现对表达式或文法的处理。我们可以使用解释器模式处理语言的解析,在设计计算机语言的时候解释器模式很有用处。...

1817
来自专栏nnngu

经典Java面试题收集

2、访问修饰符public,private,protected,以及不写(默认)时的区别?

92513

扫码关注云+社区