C++的优秀特性5:模版

(转载请注明原创于潘多拉盒子)

  C++是强类型语言,而且恐怕是强类型语言里面类型最严格的。这意味着:1. C++变量的类型在定义时就确定了;2. 该类型在后续的生命期中不会改变。比如:

int n = 0;
n = 3.14159;    // n的类型不变,仍然为int型

这样编译器产生的机器码是确定的,不需要运行时编译,比如像Python中的JIT(Just In Time)那样在代码执行过程中编译。

  但是,这给代码的可复用性带来了麻烦。比如现实中我们可能会定义一个二元关系"<",它表示“如果a < b,那么a在b前面”,或许你可以简单的理解为小于号,或者更广义的说,是a在b前面。那么你可能可以基于这个顺序的定义写一个排序函数。比如:

void sort(int* array, int len)
{
    // 某种排序实现,比如冒泡或者快速排序,就地对array中的元素进行排序
}

貌似就OK了!但仔细一看,不对!如果array是double型呢?幸好C++有重载(overload)机制,这事儿也不难:

void sort(double* array, int len)
{
    // 某种排序实现,比如冒泡或者快速排序,就地对array中的元素进行排序
}

由于是sort是完全基于 "<"运算符的,因此可以想想这两段实现代码看起来会非常像,出了中间变量的类型不同之外,其余都是相同的。甚至你可以直接拷贝粘贴,可以轻松实现float、long甚至是string的排序函数。

  不过,这很繁琐,不仅如此,还带来了维护的问题,如果代码需要改动呢?C++为解决这类问题,提供了模版机制,它允许程序员用"抽象类型"来暂时替代具体的类型,比如上述的例子可以写成:

template <typename T>
void sort(T* array, int len)
{
    // 直接实现基于抽象类型T的排序函数,这里的类型T是可以在编译时确定的
}

注意,这段代码必须实现在头文件中,不可以实现在cpp中,除非该函数仅仅由该cpp调用。有了这样一个函数的定义,在客户代码中如何调用呢?该怎么调用就怎么调用,比如:

 std::string names[100];
// load names ...
sort(names, 100);    // 编译器在此时完成模版的实例化

  除了可以定义模版函数之外,还可以定义模版类,比如标准模版库(STL)就是基于模版机制实现的C++标准库。比如你可以定义一个类Any(boost库中有完整的实现boost::any),用于封装所有的类型。这个类可以定义成这样:

class Any
{
public:
    Any() : _object(NULL), _type() {}

    Any(const Any& other) : _object(other._object), _type(other._type) {}

    Any& operator = (const Any& other)
    {
    _object = other._object;
    _type = other._type;
return *this;
} // 可以将任意类型T的对象存储在这个Any类型中   template <typename T>   Any& operator = (const T& object)   {     _holder = new T(object);     _type = typeid(T);
return *this;   } // 定义了向已知类型T的转换操作   template <typename T>   T& cast()   {     if (typeid(T) != _type)     {       throw std::exception("bad type");     }     return *reinterpret_cast<T>(_object);   } private: void* _object; std::type_info _type; };

这个类使用起来是这样的:

std::string name = "ZHANG San";
Any attribute = name;  // 这个any内部保存的类型被实例化为std::string
std::string& anotherName = name.cast<std::string>();  // 由于没有参数列表,只能在模版实例化指示部分<std::string>定义实例化类型
assert(name == anotherName);  // must be true

  或许,你需要定义一个容器类,像STL那样,保存某些元素:

template <typename T>
class Container
{
public:
    // 具体定义
};

// 客户代码
Container<double> scores;

  

C++的模版机制较复杂,实际使用中也会遇到不少问题,这里因为内容深度的限制,就不继续讨论了。

 

posted @ 2014-02-16 10:25  潘多拉盒子  阅读(371)  评论(0编辑  收藏  举报