来源:http://c.chinaitlab.com/cc/ccjq/200806/752604_2.html

--  template 的用法
    在程序设计当中经常会出现使用同种数据结构的不同实例的情况。例如:在一个程序中
    可以使用多个队列、树、图等结构来组织数据。同种结构的不同实例,也许只在数据元素
    的类型或数量上略有差异,如果对每个实例都重新定义,则非常麻烦且容易出错。那么能
    否对同种类型数据结构仅定义一次呢?答案是肯定的,C++提供的类模板(Class Template
    )就可以实现该功能。
    一、类模板
    类模板是C++提供的一种特殊机制,通过它我们可以定义一种特殊的类(称为模板类),在类
    的定义中可以包含待定的类型参数,在声明类的实例时,系统会自动根据传递的类型生成
    用户想要生成的类实例。下面是用C++实现的一个简单的模板类Clist的定义。

  1. Template <class T, int I> class CList
  2. {
  3. public:
  4. int SetItem(int Index, const T &Item);
  5. int GetItem(int Index, T &Item);
  6. private:
  7. T Buffer;
  8. }

在这里,T是类型参数,I是整型常量参数。T和I的实际值是在声明具体类实例时指定的。
    模板类的<>号内可以包括任意个类型参数和常量参数(至少要有一个参数)。类型参数和
    常量参数可以是任何合法的标准类型和用户自定义类型,包括简单类型及各种结构体。同
    其他类一样,类成员函数SetItem的实现可以在类定义内完成,也可以在类CList定义处实
    现:

  1. template<class T, int I> int CList<T, I>::SetItem(int Index, const T &Item)
  2. {
  3. if ( (Index<)||(Index>I-) )
  4.  return ; // 出错
  5. Buffer[Index]= Item ;
  6.  return ; // 成功
  7. }

值得注意的是,在类定义外完成函数实现时,必须以关键字template和类模板定义中相同
    的参数表(<>号内的)开头(上例为template<class T, int I>),并且范围分解操作符前的
    类名后应跟上模板参数名清单(上例为CList<T, I>)。另外,与非模板类不同的是,必须将
    函数实现包括在调用它的每个源文件中,使编译器能从函数实现产生代码。通常的做法是
    将模板类的函数实现也放在定义该类的头文件中,这样只需在调用的源文件中包含该头文
    件即可。
    那么,如何使用生成特定的类实例呢?我们可以像使用其他类一样来使用模板类,不过必须
    指定模板参数的值。例如采用如下声明:
    CList <int, 100> IntList;
    则使IntList成为CList类的实例,每次出现的T参数都换成int, 每次出现的I参数都换成
    100。这样,IntList类中的Buffer就是一个长度为100的整型数组,SetItem和GetItem函数
    参数是int值的引用。例:
    IntList.SetItem(0, 5); //给数组第一个元素赋为整数5
    模板类还可以像其他类一样可以定义构造函数和析构函数。下面我们以一种简单的数据
    结构——堆栈为例,来说明如何用类模板来构造通用数据结构。
    二、 利用类模板实现通用堆栈结构
    任何抽象数据结构在计算机中的实现,归根结底都只有两种方式:顺序存储(用数组实现)
    ,链式存储(用指针实现)。堆栈也不例外,按其实现方式可分为顺序栈(用数组实现)和链
    栈(用指针实现)。
    1. 通用顺序栈的实现
    因为顺序栈中的元素在空间上连续存储,栈顶的元素位置需要注明,所以构造顺序栈的模
    板类应该有这样的一些成员变量:一个待定类型和长度的数组Buffer,一个记录栈顶元素
    的数组下标的整型变量top。堆栈的基本操作主要有:入栈(Push)、出栈(Pop)、置空(Se
    tEmpty)、判断当前状态(IsEmpty)等,它们应用模板类的成员函数来实现。作为一个标准
    的类,它还应该有自己的构造函数和析构函数。具有这些功能的模板类,就可以作为一个
    通用的顺序栈来使用了。该类的定义如下:

  1. template <class T,int SIZE> class CArrayStackTemp
  2. {
  3. public:
  4. CArrayStackTemp () //缺省构造函数,构造一个空堆栈
  5. {
  6. top= -;
  7. };
  8. CArrayStackTemp (){};//析构函数
  9.  void SetEmpty (); //置空堆栈
  10.  bool IsEmpty(); //判断堆栈是否为空
  11.  bool Push(T element); //入栈
  12.  bool Pop(T& element);//出栈
  13. private:
  14. T Buffer[SIZE];
  15.  int top;
  16. };

与堆栈的基本操作相对应的成员函数的实现如下:

  1. template <class T, int SIZE> void CArrayStackTemp<T, SIZE>:: SetEmpty ()
  2. {
  3. top= -; //将栈顶指针赋 -1,并不实际清除数组元素
  4. }
  5. template <class T, int SIZE> bool CArrayStackTemp<T, SIZE>:: IsEmpty ()
  6. {
  7. return(top == -);
  8. }
  9. template <class T, int SIZE> bool CArrayStackTemp<T, SIZE>:: Push (T element
  10. )
  11. {
  12. top++;
  13. if (top>SIZE-)
  14. {
  15. top--;
  16. return false; //堆栈已满,不能执行入栈操作
  17. }
  18. Buffer[top]=element;
  19. return true;
  20. }
  21. template <class T, int SIZE> void CArrayStackTemp<T, SIZE>:: Pop (T& element
  22. )
  23. {
  24. if (IsEmpty())
  25.  return false;
  26. element =Buffer[top];
  27. top--;
  28. return true;
  29. }

根据实际需要,还可以扩充堆栈功能。例如:加入取栈顶元素、求堆栈长度等操作,其方法
    如上。
    2. 通用链栈的实现
    模板类中允许使用指针和定义自己的结构,这就为实现链式结构提供了保证。这里采用一
    个单链表来实现堆栈,栈顶指针指向链表的第一个结点,入栈和出栈均在链表的头进行。
    该模板类的定义如下:

  1. template <class T> class CLinkStackTemp
  2. {
  3.  
  4. public:
  5.  //类的缺省构造函数,生成一个空堆栈
  6. CLinkStackTemp ()
  7. {
  8. top=NULL;
  9. };
  10. ClinkStackTemp(){}; //析构函数
  11.  //定义结点结构
  12.  struct node
  13. {
  14. T
  15.   data; //入栈元素
  16.  node* next; //指向下一结点的指针
  17. };
  18.  void SetEmpty(); //置空堆栈
  19.  bool IsEmpty(); //判断堆栈是否为空
  20.  bool Push(T element); //压入堆栈
  21.  bool Pop(T& element);//弹出堆栈
  22. private:
  23.  node* top;
  24. };

该类的成员函数实现如下:

  1. template <class T> void CLinkStackTemp <T>::SetEmpty()
  2. {
  3. //释放堆栈占用的内存
  4. node* temp;
  5. while (top!=NULL)
  6. {
  7.  temp=top;
  8.  top=top->next;
  9.  delete temp;
  10. }
  11. }
  12. template <class T> bool CLinkStackTemp <T>::IsEmpty()
  13. {
  14. return (top==NULL);
  15. }
  16. template <class T> bool CLinkStackTemp <T>::Push(T element)
  17. {
  18. node* temp=new node();
  19. if (temp ==NULL)
  20.  return false ;
  21. temp->data=element;
  22. temp->next=top;
  23. top=temp;
  24. return true;
  25. }
  26. template <class T> bool CLinkStackTemp <T>::Pop(T& element)
  27. {
  28. if ( IsEmpty())
  29.  return false;
  30. node* q = top;
  31. element = top->data;
  32. top=top->next;
  33. delete q;
  34. return true;
  35. }

与顺序栈的实现略有不同,链栈不必指定栈的容量,其大小可以是近似"无限"的。为了程
    序的使用方便,我们同样可以加入一些增强的功能。
    三、 通用堆栈类的使用
    通用堆栈类的使用较为简单,堆栈类的实例就是一个可以方便使用的堆栈。对堆栈的操作
    都是通过类的成员函数来实现的。使用的具体步骤如下:
    1. 在要使用堆栈类的程序代码的文件开头包括模板类及其成员函数的定义。
    2. 类的实例化,可声明成变量,也可以声明它的指针,如:
    CArrayStackTemp <int, 100> intStack; //生成一个长度为100的int型堆栈
    //生成一个元素为Record型的堆栈,Record为自定义结构
    CLinkStackTemp <Record>* RecordStack;
    RecordStack=new CLinkStackTemp<Record>;
    应注意在定义顺序栈时,必须指定栈的大小,而链栈则不必。另外在指定指针类型和执行
    new操作时,必须对模板参数赋值,并且前后要一致。
    3. 对堆栈进行操作,如:
    intStack.Push(3); //将整数3入栈
    RecordStack.SetEmpty(); //将堆栈置空
    无论我们使用哪种堆栈类,对用户来讲都是透明的,操作起来并无差别。

C++ primer里的template用法的更多相关文章

  1. Hibernate(或其它ORM)里的inverse用法详解,内容摘自Java web轻量级开发面试教程

    本文来是从 java web轻量级开发面试教程从摘录的. Inverse的英文含义是反转,在Hibernate中用来决定是由哪方来维护两个业务实体类之间的关联关系,具体而言,就是由哪方去设置这个被外键 ...

  2. Go标准库:Go template用法详解

    本文只介绍template的语法和用法,关于template包的函数.方法.template的结构和原理,见:深入剖析Go template. 入门示例 以下为test.html文件的内容,里面使用了 ...

  3. vue里ref ($refs)用法

    ref 有三种用法: 1.ref 加在普通的元素上,用this.ref.name 获取到的是dom元素 2.ref 加在子组件上,用this.ref.name 获取到的是组件实例,可以使用组件的所有方 ...

  4. Linux编程里getopt_long_only函数用法详解

    在程序中难免需要使用命令行选项,可以选择自己解析命令行选项,但是有现成的,何必再造轮子.下面介绍使用getopt_long_only和getopt_long(两者用法差不多)解析命令行选项. 程序中主 ...

  5. ABAP里SELECT的用法汇总(转)

    通常使用Open SQL的数据查询语句SELECT将数据库条目选择到内存.一.SELECT语句:1)SELECT用于确定读取数据表中的哪些字段:2)FROM子句用于确定从哪些内表或者视图中读取数据:3 ...

  6. 【译】css动画里的steps()用法详解

    原文地址:http://designmodo.com/steps-c... 原文作者:Joni Trythall 我想你在css 动画里使用steps()会和我一样有很多困惑.一开始我不清楚怎样使用它 ...

  7. SQL里IN的用法以及优化

    1.in后条件不多,可以考虑主表建索引,或用union all 代替 2. in 和 exists的区别: 如果子查询得出的结果集记录较少,主查询中的表较大且又有索引时应该用in, 反之如果外层的主查 ...

  8. android里Toast的用法

    在活动中,可以通过findViewById()方法获取到在布局文件中定义的元素,这里我们传入R.id.button_1,来得到按钮的实例,这个值是刚才在first_layout.xml中通过andro ...

  9. KEIL里 Volatile的用法

    volatile用于防止相关变量被优化. 例如对外部寄存器的读写.对有些外部设备的寄存器来说,读写操作可能都会引发一定硬件操作,但是如果不加volatile,编译器会把这些寄存器作为普通变量处理,例如 ...

随机推荐

  1. 使用 Storyboard Segue 实作 UIViewController 的切换

    http://blog.csdn.net/mazhen1986/article/details/7791430 Storyboard 是在 iOS 5 SDK 中才出现的新名词,它其实就是原本的 Xi ...

  2. LINQ——语言级集成查询入门指南(1)

    本文主要是对语言级集成查询或简称为LINQ做一个介绍,包括LINQ是什么,不是什么,并对它在语言特性方面做一个简短的回顾,然后举一些使用LINQ的实际例子进行说明. 语言级集成查询是什么? 在我过去写 ...

  3. linux入门教程(五) Linux系统的远程登录

    首先要说一下,该部分内容对于linux初学者来讲并不是特别重要的,可以先跳过该章节,先学下一章,等学完后再回来看这一章. Linux大多应用于服务器,而服务器不可能像PC一样放在办公室,它们是放在ID ...

  4. truncate、drop、delete区别

    速度:drop>truncate>delete 1.TRUNCATE TABLE 在功能上与不带 WHERE 子句的 DELETE 语句相同:二者均删除表中的全部行.但 TRUNCATE ...

  5. MAC 上升级python为最新版本

    第1步:下载Python3.4 下载地址如下: 下载Mac OS X 64-bit/32-bit installer https://www.python.org/downloads/release/ ...

  6. lintcode :Segmemt Tree Build II

    题目 Segmemt Tree Build II The structure of Segment Tree is a binary tree which each node has two attr ...

  7. [hackerrank]Even Odd Query

    https://www.hackerrank.com/contests/w5/challenges 简单题,注意整数的0次方是1,奇数. #include <iostream> #incl ...

  8. AJAX POST请求中参数以form data和request payload形式在servlet中的获取方式

    转载:http://blog.csdn.net/mhmyqn/article/details/25561535 HTTP请求中,如果是get请求,那么表单参数以name=value&name1 ...

  9. 客户端一个http连接包含两个方向,一个是这个http连接的输入,另一个是这个http连接的输出。

    1.客户端一个http连接包含两个方向,一个是这个http连接的输入,另一个是这个http连接的输出. 利用httpclient进行ip地址和端口号连接后,http的输出端作为http请求参数设置.h ...

  10. NPOI基础入门(旧版本)

    1.常用的类与方法 工作本HSSFWorkbook 构造方法,无参表示创建一个新的工作本,可以接收一个流用于打开一个现有的工作本 方法CreateSheet(索引):创建指定索引的sheet对象 方法 ...