条款48 认识template元编程
模板元编程,元编程本质上就是将运行期的代价转移到编译期,它利用template编译生成C++源码,举下面阶乘例子:
template<unsigned N>
struct Factorial
{
enum { value = N * Factorial<N-1>::value};
};
template<>
struct Factorial<0>
{
enum {value = 1};
};
在编译期,Factorial<5>::value
就被翻译成了5 * 4 * 3 * 2 * 1,在运行期直接执行乘法即可。
使用示例如下:
Factorial<5> fac;
cout << fac.value << endl;
cout << Factorial<5>::value << endl;
元编程有何优点?
以编译耗时为代价换来卓越的运行期性能,因为对于产品级的程序而言,运行的时长远大于编译时长。
将原来运行期才能发现的错误提前到了编译期,要知道,错误发现的越早,代价越小。
元编程有何缺点?
代码可读性差,写起来要运用递归的思维,非常困难。
调试困难,元程序执行于编译期,不能debug,只能观察编译器输出的error来定位错误。
编译时间长,运行期的代价转嫁到编译期。
可移植性较差,老的编译器几乎不支持模板或者支持极为有限。
模板元编程TMP虽然有这么多的缺点,但它已经被证明是“图灵完全”的了,意思是它的威力大到足以计算任何事物。目前boost库中用到了一些TMP技术,但大部分项目还是因为TMP的一些缺点而没有广泛采用,所以这里我们只要略做了解即可。
总结
TMP可将工作由运行期转移到编译期,因而得以实现早期错误侦测或者更高的执行效率。
TMP可被用来生成“基于政策选择组合”的客户定制代码,也可以用来避免生成对某些特殊类型并不适合的代码(生成客户定制之设计模式)。