Rozważ ten przykład (pochodzący stąd ):
#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;
}
To kompiluje bez błędów na gcc9.2 ale gcc7.2 i brzękiem 10.0.0 narzekać B
nie jest kompletna. Błąd Clangs to:
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
źródło
źródło
.f()
. To ma sens; niekompletny typB
nie ma członkaf
.std::declval
nie ma już znaczenia, czy typ był kompletny, czy nie (i chyba się mylę)B
nie jest ani kompletny ani uważany za kompletny walias-declaration
.Odpowiedzi:
Źródłem błędu nie jest
std::declval
, ale niekompletny dostęp członka klasy.Aż rozdzielczość CWG1836 została połączona 2,5 roku temu, a średnia wymagana klasę być kompletne w wyrażeniu dostępu członkiem klasy (
E1.E2
).[expr.ref] / 2 w C ++ 11 :
[expr.ref] / 2 w C ++ 17 :
A klasa
alias-declaration
sama w sobie nie jest uważana za kompletnąmember-specification
.[class.mem] / 6 w C ++ 17 :
źródło
Od [declval] :
To sformułowanie jest obecne od C ++ 11 (więc nie jest możliwe, aby kompilatory były zgodne z wcześniejszym standardem)
źródło
T
powinien to być absolutnie pełny typ. Cieszę się, że sprawdziłem standard.