Betrachten Sie dieses Beispiel (von hier ):
#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;
}
Es wird ohne Fehler auf gcc9.2 kompiliert, aber gcc7.2 und clang 10.0.0 beschweren sich über die Unvollständigkeit B
. Clangs Fehler ist:
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;
~~~^
c++
language-lawyer
incomplete-type
declval
idclev 463035818
quelle
quelle
.f()
. Das macht Sinn; Der unvollständige TypB
hat kein Mitgliedf
.std::declval
es keine Rolle mehr spielt, ob der Typ vollständig ist oder nicht (und ich glaube, ich bin damit falsch)B
ist weder vollständig noch als vollständig inalias-declaration
.Antworten:
Die Fehlerquelle ist nicht
std::declval
, sondern unvollständiger Zugriff auf Klassenmitglieder.Bis zur Auflösung der Auflösung von CWG1836 vor 2,5 Jahren erforderte der Standard, dass die Klasse in einem Zugriffsausdruck für Klassenmitglieder (
E1.E2
) vollständig sein muss .[expr.ref] / 2 in C ++ 11 :
[expr.ref] / 2 in C ++ 17 :
Und eine Klasse ist nicht als abgeschlossen betrachtet in
alias-declaration
innerhalb seiner eigenenmember-specification
.[class.mem] / 6 in C ++ 17 :
quelle
Aus [Deklaration] :
Dieser Wortlaut ist seit C ++ 11 vorhanden (daher können Compiler nicht einem früheren Standard entsprechen).
quelle
T
unbedingt ein vollständiger Typ sein sollte. Ich bin froh, dass ich den Standard überprüft habe.