您想问的问题可能与这个问题重复:为什么 std::result_of 将(不相关的)函数类型作为类型参数?
让我们剖析一下:
std::result_of<F(Ts...)>::type
所以,在某个地方namespace std
,我们有一个类模板result_of<>
。它接受一个模板类型参数;即,它看起来基本上是这样的:
template<typename Foo>
struct result_of
{
typedef FOOBARBAZ type;
};
好的,所以,我们用参数实例化这个模板F(Ts...)
。这是不寻常的语法!您大概知道这Ts
是一个参数包,因此Ts...
括号内将在编译时扩展为以逗号分隔的类型列表,例如int, double, bool
. 所以我们有F(int, double, bool)
. 好的,这是一个函数类型。
正如int(char)
“函数获取char
和返回”的意思int
一样,也F(int, double, bool)
意味着“函数获取int, double, bool
和返回F
”。
“但是等等,”你说。“我以为F
已经是我的函数类型了!”
是的。F
是你的函数类型。但是期望的类型std::result_of
是,真的!,该函数类型包含在另一个函数类型中。详细说明:
typedef int (*F)(char);
typedef F G(char);
static_assert(std::is_same< std::result_of<G>::type, int >::value);
static_assert(std::is_same< std::result_of<F(char)>::type, int >::value);
static_assert(std::is_same< std::result_of<int (*(char))(char)>::type, int >::value);
上面的每一行都是完全等价的:F(char)
只是一种更美观的写作方式int (*(char))(char)
。当然,你不能总是侥幸逃脱,因为有时F
是函数类型不能从函数中返回:
typedef int F(char);
std::result_of<F(char)>; // fails to compile
正如@Simple 在评论中所写,std::result_of<F(Ts...)>::type
总是可以用不那么聪明但也不那么混乱的表达来代替
decltype( std::declval<F>() ( std::declval<Ts>()... ) )
即,“使用类型参数decltype
调用类型值的结果。这里没有古怪的高级函数类型;一切都按照您自然期望的方式工作。就个人而言,我可能会使用在我自己的代码中使用方法,只是因为它更容易理解;但我想有些人会更喜欢这种方法,因为它看起来表面上更简单,并且受到标准的祝福。每个人都有自己的。:)F
Ts...
decltype
std::result_of