C++11 模板元编程 - Traits in TLP


C++标准库STL中的type_traits文件中,已经有了比较全面的C++ trait组件,可以用来对代码做各种静态反射。

TLP库中补充了如下几个有用的trait工具,这些trait在后面介绍的TLP的sample代码中会用到。

  • __is_convertible(T, U):用于判断类型T是否可以默认转型为U类型;

  • __is_both_convertible(T, U):用于判断类型T和U之间是否可以互相转型;

  • __is_base_of(T, U):用于判断类型T是否是类型U的父类;

  • __is_built_in(T):用于判断类型T是否为C++内置类型;

  • __lambda_return(Lambda):针对一个Lambda表达式类型,计算其返回值类型;

  • __lambda_paras(Lambda):针对一个Lambda表达式类型,计算其参数类型;将所有参数类型放在一个TypeList中返回;

  • __lambda_para(Lambda,Index):针对一个Lambda表达式类型,返回其Index位置的参数的类型。如果没有参数返回NullType;

这些trait工具的实现大多在前面都已经介绍过,除过最后三个关于lambda的。

C++11引入了lambda特性,这是一个非常有用的特性,尤其对于编写框架。现实中我们经常把变化的算法交给客户自定义,然后通过语言提供的技术手段再注入给框架。相比传统使用虚接口做注入,支持lambda会让客户的代码更加简洁和紧凑。在框架中,我们可能会要提取用户注入的lambda表达式的返回值类型或者参数类型。TLP中的__lambda_return()__lambda_paras()__lambda_para()就是帮助代码完成这些事情的。它们的实现如下:

// "tlp/traits/LambdaTraits.h"

template<typename T>
struct LambdaTraits
: LambdaTraits<decltype(&T::operator())>
{
};

template<typename C, typename R, typename ... Args>
struct LambdaTraits<R (C::*)(Args...) const>
{
    using ReturnType = R;
    using ParaTypes = typename TypeList<Args...>::Result;
};

#define __lambda_return(...)  typename LambdaTraits<__VA_ARGS__>::ReturnType

#define __lambda_paras(...)   typename  LambdaTraits<__VA_ARGS__>::ParaTypes

#define __lambda_para(lambda, index) __at(__lambda_paras(lambda), __int(index))

如上,我们主要靠模板特化的模式匹配特性来萃取出我们想要的类型信息的:template<typename C, typename R, typename ... Args> struct LambdaTraits<R (C::*)(Args...) const>。我们把返回值类型保存在ReturnType,而把所有的参数类型保存在一个TypeList中:using ParaTypes = typename TypeList<Args...>::Result

可以如下测试这些lambda traits:

TEST("calculate the return type and parameter types of a lambda")
{
    template<typename T>
    void testLambdaTraits(const T&)
    {
        ASSERT_EQ(__lambda_return(T), int);
        ASSERT_EQ(__lambda_paras(T), __type_list(const int*, char));
        ASSERT_EQ(__lambda_para(T, 0), const int*);
        ASSERT_EQ(__lambda_para(T, 1), char);
        ASSERT_EQ(__lambda_para(T, 2), __null());
    }

    void run()
    {
        testLambdaTraits([](const int* x, char y){ return *x + y; });
    }
};

上面我们对lambda表达式[](const int* x, char y){ return *x + y; }进行了萃取,判断其返回类型是int,参数类型分别是const int *char

上面的测试中,之所以要定义testLambdaTraits函数,主要是为了从lambda表达式对象中提取出它的类型,然后再传给__lambda_return()等。这种通过函数模板来对具体对象或变量进行推导以获取其类型的手段,在C++11之前是一种常用技巧。由于C++11引入了decltype关键字,所以以后都不用再这么绕了!我们重新实现测试用例如下:

TEST("calculate the return type and parameter types of a lambda")
{
    void run()
    {
        auto f = [](const int* x, char y){ return *x + y; };
        using Lambda = decltype(f);

        ASSERT_EQ(__lambda_return(Lambda), int);
        ASSERT_EQ(__lambda_paras(Lambda), __type_list(const int*, char));
        ASSERT_EQ(__lambda_para(Lambda, 0), const int*);
        ASSERT_EQ(__lambda_para(Lambda, 1), char);
        ASSERT_EQ(__lambda_para(Lambda, 2), __null());
    }
};

之所以还需要run()方法,是因为TEST本质上是一个类,而auto f不能是定义为类的成员的,但是可以定义到函数里面。

关于TLP中的trait就介绍到这里,更多的关于trait的应用在后面的还会专门介绍。


Test in TLP

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容