c++ - installed - obtener<cadena> para las variantes falla bajo clang++ pero no g++
llvm 3.7 0 (1)
El siguiente código:
variant<string> x = "abc";
cout << get<string>(x) << "/n";
funciona bien bajo g ++ (versión 7.2). Sin embargo, cuando se compila en clang ++ (versión 5.0) usando libstdc ++, obtengo el siguiente error en el método de get
:
/usr/bin/../lib64/gcc/x86_64-pc-linux-gnu/7.2.0/../../../../include/c++/7.2.0/variant:238:46: fatal error: cannot cast ''std::variant<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >'' to its private base class ''std::__detail::__variant::_Variant_storage<false, std::
__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >''
return __get(std::in_place_index<_Np>, std::forward<_Variant>(__v)._M_u);
¿Es este un error de compilación, o mi código es ilegal de alguna manera?
Esto es causado por el error 31852 (y también 33222 ), cuya reproducción por cortesía de Jonathan Wakely debería parecer muy relevante:
template<typename V> auto get(V&) { }
template<typename>
class variant
{
template<typename V> friend auto get(V&);
};
int main()
{
variant<int> v{};
get(v); // error: ambiguous
}
clang no reconoce correctamente las declaraciones de amigos que tienen tipos de marcadores de posición. Que es exactamente cómo libstdc ++ implementa std::get
:
// Returns the typed storage for __v.
template<size_t _Np, typename _Variant>
constexpr decltype(auto) __get(_Variant&& __v)
{
return __get(std::in_place_index<_Np>, std::forward<_Variant>(__v)._M_u);
}
Esto accede a un miembro privado de variant
, pero esta función se declara correctamente como friend
:
template<size_t _Np, typename _Vp>
friend constexpr decltype(auto) __detail::__variant::__get(_Vp&& __v);
La implementación de libstdc ++ es válida, Clang simplemente no cree que __get
sea un friend
.