次のコード:
variant<string> x = "abc";
cout << get<string>(x) << "\n";
g++ (バージョン 7.2) では正常に動作します。ただし、libstdc++ を使用して clang++ (バージョン 5.0) でコンパイルすると、メソッドで次のエラーが発生します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);
これはコンパイラのバグでしょうか、それとも私のコードは何らかの点で違法でしょうか?
ベストアンサー1
これは、クラングバグ 31852(また、33222)、Jonathan Wakely 氏の好意により複製されたものは非常に関連性が高いと思われます。
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はプレースホルダー型を持つフレンド宣言を正しく認識しません。これはまさにlibstdc++が実装する方法です。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);
}
これはのプライベートメンバーにアクセスしますvariant
が、この関数は適切に宣言されたfriend
:
template<size_t _Np, typename _Vp>
friend constexpr decltype(auto) __detail::__variant::__get(_Vp&& __v);
libstdc++ の実装は有効ですが、clang は__get
それをとは見なしませんfriend
。