您身边的App定制专业企业--10年开发经验为您护航

18678812288
0531-88887250

c++11 function_typetraits备忘

文章编辑:2138com太阳集团 时间:2016年12月20日

获取函数或成员函数的返回类型,参数类型,参数长度,类类型。

 

函数参数列表推断基于typelist:http://www.cnblogs.com/flytrace/p/3551414.html

 

先看一个普通函数非const的特化:

 

复制代码

    template<typename R, typename... Args>

    struct function_traits<R (Args...)>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

复制代码

 使用:

 

复制代码

int testfunc1(char) { return 1; }

 

int main()

{

    bool b;

    b = std::is_same< typename function_traits<int(double)>::return_type, int>::value;

    std::cout << "is same: " << b << std::endl;

 

    b = std::is_same< typename function_traits<decltype(testfunc1)>::arg<0>::type, char>::value;

    std::cout << "is same: " << b << std::endl;

 

}

复制代码

对于各种参数类型的普通函数,都能正确推断。但重载函数的情形需要大家考虑。如下大家增加testfunc1的重载版本:

 

bool testfunc1(double, char) { return false; }

此时decltype(testfunc1)是无法编译通过的。这并不是大家的function_traits有问题。而是在没信息的情况下,decltype是无法选择testfunc1的重载版本的。除非大家在function_traits显式特化。

 

函数指针的function_traits也会遇到重载问题,如下是针对函数指针的function_traits:

 

复制代码

    template<typename R, typename... Args>

    struct function_traits<R (*)(Args...)>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

复制代码

decltye(&testfunc1)也是无法编译通过的。很显然,你自己作为编译器编辑的话,若是没有额外的信息,让你使用decltype去推断一个可重载的函数类型,你怎么能够知道用户希翼得到哪个类型?除了显示特化以提供给编译器信息外,对于函数指针,大家还可以提前转换,显式给以类型信息供编译器推断,如下:

 

int (*castfunc)(char) = &testfunc1;

b = std::is_same< typename function_traits<decltype(castfunc)>::arg<0>::type, char>::value;

std::cout << "is same: " << b << std::endl;

castfunc1在定义时得到了testfunc1正确的重载类型,因此decltype在推断castfunc时就有了信息来选择正确的类型。

 

这并不是一个程序技术问题,更算是一个逻辑问题,就好像面对有多个定义的单词,没有上下文你是无法知道它要代表什么意思的。

 

这种显示转换并不会带给大家太多困扰。因为使用function_traits的场景,基本上是一种延迟推断手段。比如得到消息后,使用泛型手法分发消息处理。而消息处理函数大家在注册的时候肯定是知道函数类型的,在注册时大家就已经可以显示转换这个函数指针而不会遇到重载问题了。直接使用decltype(testfunc1)好像在大家测试function_traits时才会遇到,嗯,另一个人也遇到了,不然我不会试验。。。

 

然而确实存在一个可能,使大家可以传入testfunc1,而不用给予完整类型信息,虽然不适用于function_traits的情况。如下:

 

http://stackoverflow.com/questions/9054703/overloaded-function-as-argument-of-variadic-template-function

 

复制代码

    template<typename ...Args>

    struct OverloadResolved

    {

            template<typename R>

            static auto static_doit( R (*f) (Args...), Args ... args ) -> R { return f(args...);}

    };

 

    template<typename ...Args>

    auto deduce(Args...) -> OverloadResolved<Args...> { return OverloadResolved<Args...>(); }

 

    template<typename T>

    struct dummy : public T { };

 

    #define doit(f, ...) ( dummy<decltype(deduce( __VA_ARGS__ ))> :: static_doit(f, __VA_ARGS__) )

复制代码

使用:

 

char aa = 'a'; double ff = 0.1;

std::cout << doit(testfunc1, aa) << "   " << doit(testfunc1, ff, aa) << std::endl;

可以看到,虽然testfunc1有2个重载版本,但仍能正确的实行testfunc1(aa)和testfunc1(ff, aa).

 

当然因为此处给出了参数信息。这是一个运行时方案,而function_traits要求大家在编译期就推断。

 

以下添加类成员函数的function_traits:

 

复制代码

template <typename R, typename T, typename... Args>

    struct function_traits<R (T::*)(Args...)>

    {

        typedef T class_type;

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

复制代码

还需要添加const,volatile修饰符的。以下是更完整的版本:

 

    template<typename T>

    struct function_traits;

 

    template<typename R, typename... Args>

    struct function_traits<R (Args...)>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template<typename R, typename... Args>

    struct function_traits<R (Args...) const>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template<typename R, typename... Args>

    struct function_traits<R (Args...) volatile>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template<typename R, typename... Args>

    struct function_traits<R (Args...) const volatile>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template<typename R, typename... Args>

    struct function_traits<R (*)(Args...)>

    {

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

 

 

 

    template <typename R, typename T, typename... Args>

    struct function_traits<R (T::*)(Args...)>

    {

        typedef T class_type;

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

 

    template <typename R, typename T, typename... Args>

    struct function_traits<R (T::*)(Args...) const>

    {

        typedef T class_type;

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template <typename R, typename T, typename... Args>

    struct function_traits<R (T::*)(Args...) volatile>

    {

        typedef T class_type;

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };

 

    template <typename R, typename T, typename... Args>

    struct function_traits<R (T::*)(Args...) const volatile>

    {

        typedef T class_type;

        typedef R return_type;

        typedef typelist<Args...> arglist;

        enum { arg_count = sizeof...(Args) };

        template<unsigned int N>

        struct arg

        {

            typedef typename at<N, arglist>::type type;

        };

    };


想要了解更多详情欢迎来电咨询18678812288
登陆网址:www.jnydkj.cn。
联系人:王经理。

XML 地图 | Sitemap 地图