免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1524 | 回复: 0
打印 上一主题 下一主题

[转贴]Template Meta Programming [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2004-06-28 00:01 |只看该作者 |倒序浏览
转自:http://www.allaboutprogram.com/about1992.html

I.背景

C++在很多人的心目中,一直是一种OO语言,而事实上,现在对C++的非OO部分的各种使用被逐渐地挖掘出来,其中最大的部分莫过于是template。STL、loki、boost,...,很多先行者为我们提供了方案,有的已经被列入C++标准的一部分。template的一个重要使用方法就是template meta programming,它利用编译器对于template的解释是静态的这一特性,让编译器在编译时做计算,可以有效的提高程序的运行速度。有关于template meta programming的记载,最早见于Erwin Unruh,他在1994年写了一个用template计算质数的程序。我希望通过这篇文章介绍一些TMP的基本技巧和应用,并且最终完成一个质数计算程序。
(阅读本文的过程中,建议你试图编译每一个给出的程序。由于所有的类只需要public成员,所以都用struct声明,但是仍然称之为类。)

II.技术

通常我们编写一个(小)程序,需要的语言支持其实不必很多,只要有顺序、选择和循环三种控制结构理论上就可以写出大多数程序了。我们先用TMP建立一个简单的语言环境。

1.打印

程序有了结果,需要有一个方式反馈给运行者,这里我们利用C++的出错信息,建立一个打印函数。要知道我们希望一切都在编译的时候结束,那么我们就必须让C++编译器在编译信息里面告诉我们,所以我们利用编译器的出错信息。当然这只是一个trick,如果你的TMP只是程序的一部分,你可以使用正常的输入输出。

template<unsigned int value>;
struct print
{
        static const unsigned int result = (unsigned char*)value;
};

这个类,每当别人引用到它的result的时候,编译器就会打印出错信息,因为一个unsigned int是不能隐式的转成一个unsigned char*的。譬如下面这段程序

template<unsigned int value>;
struct print
{
        static const unsigned int result = (unsigned char*)value;
};

unsigned int test1 = print<77>;::result;
unsigned int test2 = print<123>;::result;

在我的Dev C++里,会输出
main.cpp: In instantiation of `print<77>;':
main.cpp:7:   instantiated from here
main.cpp:4: invalid conversion from `unsigned char*' to `unsigned int'
main.cpp: In instantiation of `print<123>;':
main.cpp:8:   instantiated from here
main.cpp:4: invalid conversion from `unsigned char*' to `unsigned int'

这个输出虽然不是很好看,但也算是差强人意。

2.选择

Andrei Alexanderescu在他的大作Modern C++ Design里面使用过一个类,可以根据bool的值选择不同的类型。今天我们要写的一个是根据bool的值选择不同的整数。

template<bool condition, unsigned int value1, unsigned int value2>;
struct template_if
{
        static const unsigned int result = value1;
};

template<unsigned int value1, unsigned int value2>;
struct template_if<false, value1, value2>;
{
        static const unsigned int result = value2;
};

这里用到了模板的特化,如果你对这个不熟悉,那么大致可以这样理解:第一个template_if的定义告诉编译器,“一般的”template_if,会选择第一个值作为结果。第二个template_if告诉编译器,如果第一个参数是false的话,我们就使用第二个值(第三个参数)作为结果。下面这段代码演示了template_if的用法。

template<unsigned int value>;
struct print
{
        static const unsigned int result = (unsigned char*)value;
};

template<bool condition, unsigned int value1, unsigned int value2>;
struct template_if
{
        static const unsigned int result = value1;
};

template<unsigned int value1, unsigned int value2>;
struct template_if<false, value1, value2>;
{
        static const unsigned int result = value2;
};

template<unsigned int value>;
struct print_if_77
{
        static const unsigned int result = template_if<value == 77 , print<value>;::result , 0>;::result;
};

unsigned int test1 = print_if_77<77>;::result;
unsigned int test2 = print_if_77<123>;::result;

如果你去编译这段代码的话,你会发觉77和123都被打印出来了,虽然错误信息不一样,但是这不是我们想要的结果。为什么呢?很遗憾,对C++编译器来说,template_if<true, 1, 100>;和template<true, 1, 200>;是两个不同的类,虽然后一个参数的值我们并不关心,但是编译器必须在template初始化的时候,给出所有的参数,这就导致它会去计算print<value>;::result,当然,计算的结果就是报错。也就是说,因为编译器要计算这个值才导致了我们的print不可用,要解决这个问题,有两个方法:或者让编译器不计算这个值,或者让编译器在某些情况下可以计算出正确的值。

方法一可以让编译器不计算这个值,通过修改template_if,我们传入两个不同的类,而不是unsigned int。
首先修改print,加一个新的类dummy_print:
template<unsigned int value>;
struct print
{
        static const unsigned int result = (unsigned char*)value;
};

template<unsigned int value>;
struct dummy_print
{
        static const unsigned int result = value;
};

接着,加入一套对类型进行选择的模板:

template<bool condition, typename T1, typename T2>;
struct template_if_type
{
        static const unsigned int result = T1::result;
};

template<typename T1, typename T2>;
struct template_if_type<false, T1, T2>;
{
        static const unsigned int result = T2::result;
};

这样原先的程序就变成:

template<unsigned int value>;
struct print
{
        static const unsigned int result = (unsigned char*)value;
};

template<unsigned int value>;
struct dummy_print
{
        static const unsigned int result = value;
};

template<bool condition, typename T1, typename T2>;
struct template_if_type
{
        static const unsigned int result = T1::result;
};

template<typename T1, typename T2>;
struct template_if_type<false, T1, T2>;
{
        static const unsigned int result = T2::result;
};

template<unsigned int value>;
struct print_if_77
{
        static const unsigned int result = template_if_type<value == 77 ,
        print<value>; , dummy_print<value>; >;::result;
};

unsigned int test1 = print_if_77<77>;::result;
unsigned int test2 = print_if_77<123>;::result;

现在的“运行结果”非常正确。

方法二可以让编译器在某些情况下计算出正确的值,我们加一套新的模板:

template<bool condition, unsigned int value>;
struct print_if
{
        static const unsigned int result = (unsigned char*)value;
};

template<unsigned int value>;
struct print_if<false, value>;
{
        static const unsigned int result = value;
};

原先的程序变为:

template<bool condition, unsigned int value>;
struct print_if
{
        static const unsigned int result = (unsigned char*)value;
};

template<unsigned int value>;
struct print_if<false, value>;
{
        static const unsigned int result = value;
};

template<unsigned int value>;
struct print_if_77
{
        static const unsigned int result = print_if<value == 77 , value>;::result;
};

unsigned int test1 = print_if_77<77>;::result;
unsigned int test2 = print_if_77<123>;::result;

输出也是正确的。

这两种方案,我个人倾向于后者,因为其实我们一定是要做一次判断的,并且这次判断一定会添加新的类,那么还是print_if的解决方案比较直观。

3. 循环
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP