考虑以下示例(来自此处):
#include <type_traits>
#include <iostream>
template <typename U>
struct A {
};
struct B {
template <typename F = int>
A<F> f() { return A<F>{}; }
using default_return_type = decltype(std::declval<B>().f());
};
int main()
{
B::default_return_type x{};
std::cout << std::is_same< B::default_return_type, A<int>>::value;
}
它在gcc9.2上编译时没有错误,但gcc7.2和clang 10.0.0抱怨B
未完成。lang的错误是:
prog.cc:11:58: error: member access into incomplete type 'B'
using default_return_type = decltype(std::declval<B>().f());
^
prog.cc:7:8: note: definition of 'B' is not complete until the closing '}'
struct B {
^
prog.cc:16:8: error: no type named 'default_return_type' in 'B'
B::default_return_type x{};
~~~^
prog.cc:17:35: error: no member named 'default_return_type' in 'B'
std::cout << std::is_same< B::default_return_type, A<int>>::value;
~~~^
@MSalters我也这么认为,但是这里真正的问题是什么?我假设一旦您从中获得实例,
—
idclev 463035818
std::declval
类型是否完整就不再重要了(我想我是错的)
[expr.ref] / 2(C ++ 11)关于类成员访问的说法:“对于第一个选项(点),第一个表达式应具有完整的类类型”。并且
—
语言律师,
B
在中既不完整也不认为完整alias-declaration
。
@LanguageLawyer我没有找到您引用的句子,但只有“该类的类型是完整的,除非该类的定义中出现该类成员的访问权限”
—
idclev 463035818
@LanguageLawyer好的,那么我同意我的解释已关闭,并且自c ++ 11以来似乎有所改变,这使得在较新的标准中可以实现上述要求,但在c ++ 11中却没有。您介意写答案吗?
—
idclev 463035818
.f()
。这就说得通了; 不完整类型B
没有成员f
。