get <string>表示变体在clang ++下失败但不是g ++

And*_*ner 25 c++ variant clang++ c++17

以下代码:

variant<string> x = "abc";
cout << get<string>(x) << "\n";
Run Code Online (Sandbox Code Playgroud)

在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);
Run Code Online (Sandbox Code Playgroud)

这是编译器错误,还是我的代码以任何方式违法?

Bar*_*rry 24

这是由clang bug 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 
}
Run Code Online (Sandbox Code Playgroud)

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);
}
Run Code Online (Sandbox Code Playgroud)

这访问了一个私有成员variant,但是这个函数被正确声明为friend:

template<size_t _Np, typename _Vp>
friend constexpr decltype(auto) __detail::__variant::__get(_Vp&& __v);
Run Code Online (Sandbox Code Playgroud)

libstdc ++的实现是有效的,clang只是不认为__get是一个friend.

  • 更改基类的访问权限是过度的,只需将`public:using _Base :: _ M_u;`添加到`variant`类定义的末尾即可.我已经在https://gcc.gnu.org/r258854上完成了 (7认同)
  • 我认为最好和相当安全的解决方法是在开发过程中使用libstdc ++的变体头文件的修改版本.您需要更改的是注释掉友元声明并公开派生自__detail :: _ variant :: _ Variant_base(它允许__get访问底层变体存储).我使用它来扼杀大量的错误,同时使用GCC作为主要编译器,并没有发现任何不良副作用. (3认同)