模板之长整数映射为型别

template <int v>
struct Int2Type
{
	enum { value = v };
};

Int2Type会根据引数所得的不同数值产生不同的型别。这是因为“不同的 template 的具现体”本身是“不同的类型”。因此 Int2Type<0> 不同于 Int2Type<1>,以此类推。用来产生型别的那个数值是一个枚举值。

符合下列两个条件便可使用 Int2Type:

1 有必要根据某个编译期常数调用一个或数个不同的函数

2 有必要在编译器实施“分派”(dispatch)


执行期的分派可使用 if-else 或 switch 语句。但 if-else 语句要求每一个分支都得编译成功,即使该条件测试在编译期才知道。

假设你设计一个泛型容器 NiftyContainer,他将元素型别参数化:

template <typename T, bool isPolymorphic>
class NiftyContainer
{
public:
	void DoSomething()
	{
		T *pSomobject = ..;
		if (isPolymorphic)
		{
			T *pNewObj = pSomobject->Clone(); // virtual function :Clone
			//...polymorphic algorithm...
		}
		else
		{
			T *pNewObj = new T(*pSomobject); // copy constructor,no virtual function
			// ... non-polymorphic algorithm 
		}
	}
};
编译器很勤劳的编译 if-else 的每个分支,如果你想通过 isPolymorphic 来编译自己选择的代码块是不能实现的。

例如 isPolymorphic 为 false 你并未定义虚函数 Clone ,这时编译不会成功。即使我们定义了虚函数 Clone 和 Copy 构造,编译会生成 if - else 所有分支代码,这很显然不会减少生成的代码量。

通过 Int2Type 可以提供一个方案。

template <typename T, bool isPolymorphic>
class  NiftyContainer
{
private:
	void DoSomething(T *pObj, Int2Type <true>)
	{
		T *pNewObj = pObjt->Clone();
		//...polymorphic algorithm...
	}
	void DoSomething(T *pObj, Int2Type<false>)
	{
		T *pNewObj = new T(*pObj); // copy constructor
		// ... non-polymorphic algorithm 
	}
public:
	void DoSomething(T *pObj)
	{
		DoSomething(pObj, Int2Type<isPolymorphic>()); // 注:传入 Int2Type<isPolymorphic>() 临时对象
	}
};

编译器不会去编译一个未被用到的 template 函数。根据 isPolymorphic 值,编译期会选择编译指定的函数。

例如  isPolymorphic 为 true,编译器选择编译 DoSomething(T *pObj, Int2Type<true>) 函数,而不会编译 DoSomething(T *pObj, Int2Type<false>) 。

class MyClass {};

int main()
{
	NiftyContainer<MyClass, false> container;
	MyClass myObj;
	container.DoSomething(&myObj);

	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值