其他分享
首页 > 其他分享> > decltype和declval的用法

decltype和declval的用法

作者:互联网

1. decltype是c++11以后出现的一个新的关键字,是用来萃取表达式或者变量或者函数返回值的类型的。

具体用法可以参考官网:https://en.cppreference.com/w/cpp/language/decltype

2. declval是c++11中的一个模板函数,原型如下:

具体用法可以参考官网:https://zh.cppreference.com/w/cpp/utility/declval

template<class T>
typename std::add_rvalue_reference<T>::type declval() noexcept;

将任意类型 T 转换成引用类型,令在 decltype 表达式中不必经过构造函数就能使用成员函数。

3.使用案例

#include <type_traits>
#include <utility>

int func(int){return 0;} //普通函数
typedef int(*pFunc)(); //函数指针
typedef int(&refFunc)(); //函数引用
struct ClassFunc
{
    char operator()(int) { return '0'; }
    int operator()(bool) { return 0; }
    int test() { return 0; }
    float test(int) { return 0; }
    static double test(float) { return 0; }

    float MyFunc(int) { return 0; }
    static int MyTest(int) { return 0; }
};

void main()
{
    int a;
    int b;
    decltype(a) c; //int
    //双层括号的,则解析为int&
    decltype((a)) cc = c; //int&
    decltype(a + b) d; //int

    //萃取一个有参函数的返回值时,需给参数传值,不能写成func(int)
    decltype(func(0)) e; //int

    //这里萃取到的是一个函数指针,没有理解
    decltype(pFunc()) f;//int(*)();

    auto lambdaFunc = []()->int {return 0; };
    /*
    这个必须还要把lambdaFunc赋值给ff,否则就报错
    无法引用 "lambda []()->int"的默认构造函数--它是已删除的函数,无法构造lambda实例
    */
    decltype(lambdaFunc) ff= lambdaFunc;

    //类构造一个匿名对象,然后萃取该匿名对象的类型
    decltype(ClassFunc()) g; //ClassFunc

    /*萃取ClassFunc::operator()(int)的返回类型,通过调用成员函数的方式来触发萃取*/
    decltype(ClassFunc()(0)) h; //char
    decltype(ClassFunc()(false)) i; //int
    decltype(ClassFunc().test()) j; //int
    decltype(ClassFunc().test(0)) k; //float
    //因为是静态成员函数,所以可以直接通过类的方式来调用,从而萃取返回类型
    decltype(ClassFunc::test(1.2f)) m; //double
    decltype(ClassFunc().test(1.2f)) n; //double

    /*当ClassFunc的构造函数为私有的时候,那么是不能通过构造一个对象去调用函数,然后萃取返回类型的。
    此时需要 declval 来转换一个引用类型,然后调用类成员函数。
    */
    decltype(std::declval<ClassFunc>().test(0)) p; //float
}

 

标签:return,declval,int,用法,ClassFunc,test,decltype,萃取
来源: https://blog.csdn.net/c1s2d3n4cs/article/details/115354389