美文网首页
C++11 模板元编程 - 惰性

C++11 模板元编程 - 惰性

作者: MagicBowen | 来源:发表于2016-09-16 08:29 被阅读506次

C++对模板的具现化采用尽量惰性的原则。只有当你使用了模板的内部定义,编译器才会为模板生成对应的定义。

所以对于元函数,当你不访问内部的Result对其求值,编译器是不会为其做计算的。因此我们可以把一个元函数当做运行期函数指针一样进行传递,直到我们需要的时候再对其求值。

对于惰性,我们来看下面这个例子:

template<typename T, bool isClonable>
struct Creator
{
    static T* create(const T* instance)
    {
        if(isClonable)
        {
            return instance->clone();
        }
        else
        {
            return new T(*instance);
        }
    }
};

如上我们希望有一个工厂类,用来创建入参T类型的对象。如果T支持clone方法,则采用从一个现有对象clone出新对象,否则调用拷贝构造函数new出来一个新对象。Creator的第二个参数isClonable用来指示前一个参数T是否支持clone。

遗憾的是,上述代码是不能工作的。当我们传递一个不支持clone的类形T进去,即使我们将isClonable设为false,编译器也会对create函数进行完整编译,会报告T缺少clone方法。

struct UnclonableObject
{
    UnclonableObject(){}
    UnclonableObject(const UnclonableObject&) {}
};

Creator<UnclonableObject, false> creator;

然而当我们写出上述代码进行编译,发现却能编译通过!原因是C++编译器的惰性特征做了手脚,此时它没有看到任何人调用Creator<UnclonableObject, false>的create方法,所以并未生成该方法。

一旦我们写出如下代码,就会和我们最初预想的一致:编译失败,编译器告诉我们UnclonableObject中没有clone方法。

UnclonableObject object;
UnclonableObject* newObject = creator.create(&object);

解决该问题的方式很简单,就是把使用if做运行期分支选择的实现转换成使用编译期的分支选择元函数__if()来实现。

template<typename T>
struct ClonableCreator
{
    static T* create(const T* instance)
    {
        return instance->clone();
    }
};

template<typename T>
struct UnclonableCreator
{
    static T* create(const T* instance)
    {
        return new T(*instance);
    }
};

template<typename T, bool isClonable> using Creator = __if(__bool(isClonable), ClonableCreator<T>, UnclonableCreator<T>);

由于模板元编程的惰性特征,__if()元函数会根据第一个入参的bool值,对后面的两个参数中的一个进行求值。因此当我们调用Creator<UnclonableObject, false>时,__if(__bool(false), ClonableCreator<T>, UnclonableCreator<T>)只会对UnclonableCreator<T>具现化,所以没有再出现之前的编译错误。

利于惰性求值,在编译期选择性的做类型具现化,是模板元编程非常有用的特性之一。


鸭子类型

返回 C++11模板元编程 - 目录

相关文章

  • C++11 模板元编程 - 惰性

    C++对模板的具现化采用尽量惰性的原则。只有当你使用了模板的内部定义,编译器才会为模板生成对应的定义。 所以对于元...

  • C++11 模板元编程 - 元编程

    从本节开始我们将模板元编程当做一门独立的函数式语言来讨论它的方方面面。 所谓元编程,就是指可以产生程序的程序。由于...

  • C++11 模板元编程 - 模板元编程的应用

    本节开始我们通过使用C++模板元编程去解决一些实际问题,来展示模板元编程针对现实问题的使用方法和设计技巧。本节中的...

  • C++11 模板元编程 - 模板递归

    模板可以被递归调用,在模板递归的过程中,可以执行前面我们提到的两种编译期计算:数值计算和类型计算。 下面我们用模板...

  • C++11 模板元编程 - 元函数

    我们继续演进前面那个无聊的类型计算的例子,来得出元函数的定义。 前面我们实现了PointerOf,它对于传进的任意...

  • C++11 模板元编程 - 后记

    当1994年,Erwin Unruh在C++标准委员会上演示了通过C++模板在编译期计算素数的程序后,C++模板元...

  • C++11 模板元编程 - 前言

    熟悉C++的程序员都知道,C++是一门多范式编程语言,支持面向过程、面向对象、泛型编程以及函数式编程范式。然而提到...

  • C++11 模板元编程 - 目录

    前言 模板的基础知识模板的类型参数模板的默认参数模板的模板参数模板的特化模板的非类型参数模板的编译期计算数值计算类...

  • C++11 模板元编程 - 递归

    函数式语言依赖模式匹配和递归完成类似命令式语言里分支选择和循环迭代的功能。模板元编程中可以完成模式匹配的两种方式上...

  • C++11 模板元编程 - TypeList

    对函数式编程来说,list是其中最基础也是最重要的数据结构。通过list可以轻易地构造出tree,map等复杂数据...

网友评论

      本文标题:C++11 模板元编程 - 惰性

      本文链接:https://www.haomeiwen.com/subject/zgzbettx.html