版权归原作者所有,如有侵权,请联系我们

[科普中国]-模板元编程

科学百科
原创
科学百科为用户提供权威科普内容,打造知识科普阵地
收藏

模板元编程是一种元编程技术,编译器使用模板产生暂时性的源码,然后再和剩下的源码混合并编译。这些模板的输出包括编译时期常量、数据结构以及完整的函数。如此利用模板可以被想成编译期的运行。这种技术被许多语言使用,最为知名的当属C++,其他还有Curl、D、Eiffel,以及语言扩展,如Template Haskell。

模板元编程的构成要素使用模板作为元编程的技术需要两阶段的操作。首先,模板必须被定义;第二,定义的模板必须被实体化才行。 模板的定义描述了生成源码的一般形式,而使实体化则导致了某些源码的组合根据该模板而生成。

模板元编程是一般性地图灵完全(Turing-complete),这意味着任何可被电算软件表示的运算都可以透过模板元编程以某种形式去运算。

模板与宏(macros)是不同的。所谓宏只不过是以文字的操作及替换来生成代码,虽然同为编译期的语言特色,但宏系统通常其编译期处理流的能力(compile-time process flow abilities)有限,且对其所依附之语言的语义及类型系统缺乏认知(一个例外是LISP的宏)。

模板元编程没有可变的变量——也就是说,变量一旦初始化后就不能够改动。因此他可以被视为函数式编程(functional programming)的一种形式。

使用模板元编程模板元编程的语法通常与一般的程序语法迥异,他有其实际的用途。一些使用模板元编程的共同理由是为了实现泛型编程(generic programming)或是展现自动编译期最优化,像是只要在编译期做某些事一次就好,而无需每次程序运行时去做。

编译期类别生成以下将展示究竟何谓"编译期程序设计"。阶乘是一个不错的例子,在此之前我们先来回顾一下一般C++中阶乘函数的递归写法:

int factorial(int n) { if (n == 0) return 1; return n * factorial(n - 1);}void foo(){ int x = factorial(4); // == (4 * 3 * 2 * 1 * 1) == 24 int y = factorial(0); // == 0! == 1}以上的代码会在程序运行时决定4和0的阶乘。

现在让我们看看使用了模板元编程的写法,模板特化提供了"递归"的结束条件。这些阶乘可以在编译期完成计算。以下源码:

template struct Factorial { enum { value = N * Factorial::value };};template struct Factorial { enum { value = 1 };};// Factorial::value == 24// Factorial::value == 1void foo(){ int x = Factorial::value; // == 24 int y = Factorial::value; // == 1}代码如上在编译时期计算4和0的阶乘值,使用该结果值仿佛他们是预定的常量一般。

虽然从程序功能的观点来看,这两个版本很类似,但前者是在运行期计算阶乘,而后者却是在编译期完成计算。 然而,为了能够以此方式使用模板,编译器必须在编译期知道模板的参数值,也就是Factorial::value只有当X在编译期已知的情况下才能使用。换言之,X必须是常量(constant literal)或是常量表示式(constant expression),像是使用sizeof运算符。1

编译期代码最优化以上阶乘的示例便是编译期代码最优化的一例,该程序中使用到的阶乘在编译期时便被预先计算并作为数值常量植入运行码当中,节省了运行期的经常开销(计算时间)以及存储器足迹(memory footprint)。

编译期循环展开(loop-unrolling)是一个更显著的例子,模板元编程可以被用来产生n维(n-dimensional)的向量类别(当然n必须在编译期已知)。与传统n维向量比较,他的好处是循环可以被展开,这可以使性能大幅度提升。考虑以下例子,n维向量的加法可以被写成:

template Vector& Vector::operator+=(const Vector& rhs) { for (int i = 0; i