泛型编程——模板及模板的特例化

1. 简介

  面向对象编程(OOP)和泛型编程都能处理在编写程序时不知道类型的情况。不同之处在于:OOP能处理类型在程序运行之前都未知的情况;而泛型编程中,在编译时就能获知类型了。
   模板是C++中 泛型编程的基础。一个模板就是一个创建类或函数的蓝图或者说公式。当使用一个vector这样的泛型类型,或者find这样的泛型函数。这种转换发生在编译时。
  编写单一模板,使之对任何可能的模板实参都是最合适的,都能实例化,这并不总是能办到。在某些情况下,通用模板的定义对特定类型是不合适的:通用定义可能编译失败或做得不正确。其他时候,我们也可以利用某些特定知识来编写更高效的代码,而不是从通用的模板 实例化。当我们不能(或不希望)使用模板版本时,可以定义类或函数模板的一个特例化版本。
  在这里我们主要说说 模板模板的特例化

2. 函数模板

  我们可以定义一个通用的函数模板,而不是为每个类型都定义一个新函数。一个函数模板就是一个公式,可用来生成针对特定类型的函数版本。
  假如我们希望编写一个函数来比较两个值,并指出第一个值是小于、等于或大于第二个值。我们的一种方案是定义多个重载函数:
// 如果两个值相等,返回0,如果v1小于v2返回-1,如果v1大于v2返回1

int compare(const std::string &v1, const std::string &v2)
{
	if (v1 < v2)
		return -1;

	if (v1 > v2)
		return 1;

	return 0;
}

int compare(const double &v1, const double &v2)
{
	if (v1 < v2)
		return -1;

	if (v1 > v2)
		return 1;

	return 0;
}
  这两个函数几乎是相同的,唯一的差异就是参数的类型,函数体则完全一样。
  如果对每种希望比较的类型都不得不重复定义完全一样的函数体,是非常繁琐且容易出错的。更麻烦的是在编写程序的时候,我们就要确定可能要compare的所有类型。
  对此,我们的解决方案是定义一个通用的函数模板,而不是为每个类型都定义一个新函数。一个函数模板就是一个公式,可用来生成针对特定类型的函数版本。compare的模板版本可能像下面这样:

templete
int compare(const T &v1, const T &v2)
{
	if (v1 < v2)
		return -1;

	if (v1 > v2)
		return 1;

	return 0;
}
  当我们调用一个函数模板时,编译器用函数实参来为我们推断模板实参。编译器用推断出的模板来为我们实例化一个特定版本的函数。
std::cout << compare(1, 0) << std::endl;	//实例化出——int compare(const int &v1, const int &v2)

std::cout << compare(str1, str2) << std::endl;	//实例化出——int compare(const std::string &v1, const std::string &v2) 
						//其中std::string str1, str2;
  看上去我们的函数模板似乎没什么问题,但是我们定义的函数的通用模板并不适合字符指针这种特定的情况。我们希望的是通过compare调用strcmp比较两个字符指针而非比较指针值。
  为了处理字符指针,可以为我们的compare函数模板定义一个模板特例化版本。一个特例化版本就是模板的一个独立的定义,在其中一个或多个模板参数被指定为特定的类型。
  当我们特例化一个函数模板时,必须为原模板中的每个模板参数都提供实参。
// compare 的特殊版本,处理字符数组的指针
templete<>
int compare(const char *const &p1, const char * const &p2)
{
	return strcmp(p1, p2);
}
  当定义函数模板的特例化版本时,我们本质上接管了编译器的工作。即,我们为原模板的一个特殊实例提供了定义。特例化的本质是实例化一个模板,而非重载它。因此,特例化不影响函数匹配。
  除了定义类型参数,还可以在模板中定义非类型参数。一个非类型参数表示一个值而非一个类型。我们通过一个特定的类型名而非关键字class或typename来指定费类型参数。
  当一个模板被实例时,非类型参数被一个用户提供的或编译器推断出的值所代替。这些值必须是常量表达式,从而允许编译器在编译时实例化模板。
  例如,我们可以编写一个compare版本处理字符串字面常量。这种字面常量是const char的数组。由于不能拷贝一个数组,所以我们将自己的参数定义为数组的引用。由于我们希望能比较不同长度的字符串字面常量,因此为模板定义了两个非类型的参数。第一个模板参数表示第一个数组的长度,第二个参数表示第二个数组的长度。
templete
int compare(const char (&p1)[N], const char (&p2)[M])
{
	return strcmp(p1, p2);
}

当我们以下面的方式调用compare时:
compare("hi", "mom")

编译器会使用字面常量的大小来代替N和M,从而实例化模板。编译器会在一个字符串字面常量的末尾插入一个空字符作为终结符。
编译器实例化出来如下版本:
int compare(const char (&p1)[3], const char (&p2)[4])

3. 类模板

  类模板是用来生成类的蓝图。与函数模板的不同之处是,编译器不能为类模板推断参数类型。为了使用类模板,我们必须在模板名后的尖括号中提供额外信息——用来代替模板参数的模板实参列表。
  与函数模板不同,类模板分为特例化和部分特例化。也就是说我们可以只指定一部分而非所有模板参数,或是参数的一部分而非全部特性。
3.1 类模板特例化
  关于类模板及类模板的特例化,我们以boost库中的时间工具progress_timer作为例子。为此我们先来看看progress_timer的实现。
class progress_timer : public timer, private noncopyable
{
  
 public:
  explicit progress_timer( std::ostream & os = std::cout )
     // os is hint; implementation may ignore, particularly in embedded systems
     : timer(), noncopyable(), m_os(os) {}
  ~progress_timer()
  {
  //  A) Throwing an exception from a destructor is a Bad Thing.
  //  B) The progress_timer destructor does output which may throw.
  //  C) A progress_timer is usually not critical to the application.
  //  Therefore, wrap the I/O in a try block, catch and ignore all exceptions.
    try
    {
      // use istream instead of ios_base to workaround GNU problem (Greg Chicares)
      std::istream::fmtflags old_flags = m_os.setf( std::istream::fixed,
                                                   std::istream::floatfield );
      std::streamsize old_prec = m_os.precision( 2 );
      m_os << elapsed() << " s\n" // "s" is System International d'Unites std
                        << std::endl;
      m_os.flags( old_flags );
      m_os.precision( old_prec );
    }

    catch (...) {} // eat any exceptions
  } // ~progress_timer

 private:
  std::ostream & m_os;
};
  progress_timer是boost库提供的一个计时小工具,它继承自timer,会在析构时自动输出时间,但是从源码上看它的对外输出精度只有小数点后两位,即只精确到百分之一秒。但是我们知道timer在Win32平台最小精度为毫秒,在Linux平台最小精度为微秒。为了达到更高的精度,我们用模板技术来编写一个新类。
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  my_progress.h
*Author:  Huangjh
*Version:
*Date:  2018-01-23
*Description:  更高精度的计数器(boost::progress_time)模板实现
*Others:
**********************************************************************************/
#ifndef _MY_PROGRESS_H
#define _MY_PROGRESS_H

#include 
#include 

//使用模板实现的更高精度的progress_timer
template
class my_progress_timer : public boost::timer
{
public:

	explicit my_progress_timer(std::ostream & os = std::cout)
		:timer()
		,m_os(os) 
	{
	
	}

	~my_progress_timer()
	{
		try
		{
			std::istream::fmtflags old_flags = m_os.setf(std::istream::fixed,
				std::istream::floatfield);
			std::streamsize old_prec = m_os.precision(N);
			m_os << elapsed() << " s\n" 
				<< std::endl;
			m_os.flags(old_flags);
			m_os.precision(old_prec);
		}

		catch (...) {} 
	} // ~progress_timer

private:
	std::ostream & m_os;
};

//我们提供一个模板的特例化,对于精度为2的直接继承progress_timer
template<>
class my_progress_timer<2> : public boost::progress_timer
{};

#endif	//#ifndef _MY_PROGRESS_H
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  main.cpp
*Author:  Huangjh
*Version:
*Date:  2018-01-23
*Description:  测试用例
*Others:
**********************************************************************************/
#include 
#include "my_progress.h"

using namespace std;

int main(void)
{
	{
		my_progress_timer<3> myTimer;
	}

	return 0;
}
  运行结果如下所示:
0.000 s

  上面是一个非类型参数的模板,更常用的是类型模板,就如同我们经常使用的容器就是一个很好的例子了。
template
class myContainer
{
public:

	.....	

private:
	T m_Type;
	.....
};
3.2 类模板偏特化
  关于类模板的偏特化就比其他的复杂了。在使用形式上就有多钟形式。但是类模板偏特化本质上都是指定部分类型。
  我们先来看看最普通的情形:
//有二个参数的模板
template
class A { ... };

//偏特化版本,指定其中一个参数的类型
template
class A { ... };
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  template.h
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化
*Others:
**********************************************************************************/
#ifndef _TEMPLATE_H
#define _TEMPLATE_H

#include 

template
class CMyTemplate
{
public:
	void Display(void)
	{
		std::cout << "类模板普通版本:" << std::endl;
	}
};

template
class CMyTemplate
{
public:
	void Display(void)
	{
		std::cout << "类模模板偏特化版本:" << std::endl;
	}
};

#endif	//#ifndef _TEMPLATE_H
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  template.h
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化
*Others:
**********************************************************************************/
#include 
#include "template.h"

using namespace std;

int main(void)
{
	CMyTemplate c1Template;
	CMyTemplate c2Template;

	c1Template.Display();
	c2Template.Display();

	return 0;
}

  运行结果如下所示:
类模板普通版本:
类模模板偏特化版本:
  在来看看,使用的比较多的形式:
//类模板
template
class A { ... };

//只接受指针类型的偏特化版本
template
class A { ... };

//只接受引用类型的偏特化版本
template
class A { ... };
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  template.h
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化
*Others:
**********************************************************************************/
#ifndef _TEMPLATE_H
#define _TEMPLATE_H

#include 

//类模板
template
class CTemplate
{
public:
	void Display(void)
	{
		std::cout << "模板的普通版本" << std::endl;
	}
};

//只接受指针类型的偏特化版本
template
class CTemplate
{
public:
	void Display(void)
	{
		std::cout << "只接受指针类型的偏特化版本" << std::endl;
	}
};

//只接受指针类型的偏特化版本
template
class CTemplate
{
public:
	void Display(void)
	{
		std::cout << "只接受引用类型的偏特化版本" << std::endl;
	}
};

#endif	//#ifndef _TEMPLATE_H
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  main.cpp
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化的测试用例
*Others:
**********************************************************************************/
#include "template.h"

int main(void)
{
	CTemplate cTemplate1;
	CTemplate cTemplate2;
	CTemplate cTemplate3;

	cTemplate1.Display();
	cTemplate2.Display();
	cTemplate3.Display();

	return 0;
}
  运行结果:
模板的普通版本
只接受指针类型的偏特化版本
只接受引用类型的偏特化版本
  最后,还有一种形式:
//类模板
template
class A { ... };

//只接受同T实例化的vector的模板参数
template
class A> { ... }; 
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  template.h
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化
*Others:
**********************************************************************************/
#ifndef _TEMPLATE_H
#define _TEMPLATE_H

#include 
#include 

//类模板
template
class CTemplate
{
public:
	void Display(void)
	{
		std::cout << "模板的普通版本" << std::endl;
	}
};

//只接受T实例化的vector的模板参数
template
class CTemplate>
{
public:
	void Display(void)
	{
		std::cout << "只接受T实例化的vector的模板参数" << std::endl;
	}
};

#endif	//#ifndef _TEMPLATE_H
/*********************************************************************************
*Copyright(C),Your Company
*FileName:  main.cpp
*Author:  Huangjh
*Version:
*Date:  2018-01-24
*Description:  模板及特例化的测试用例
*Others:
**********************************************************************************/
#include 
#include "template.h"

using namespace std;

int main(void)
{
	CTemplate cTemplate1;
	CTemplate> cTemplate2;
	CTemplate> cTemplate3;

	cTemplate1.Display();
	cTemplate2.Display();
	cTemplate3.Display();

	return 0;
}
  运行结果:
模板的普通版本
模板的普通版本
只接受T实例化的vector的模板参数

4. 总结

  关于模板这边只是简单的进行描述,关于模板的用途。我觉得也是很难去总结出来的。更多的时候在遇到的时候在深入学习。比如:查看STL源码时。

你可能感兴趣的:(c++)