std::reference_wrapper,构造函数实现解释



我一直在尝试理解std::reference_wrapper的实现,如下所示:

namespace detail {
template <class T> constexpr T& FUN(T& t) noexcept { return t; }
template <class T> void FUN(T&&) = delete;
}
 
template <class T>
class reference_wrapper {
public:
  // types
  typedef T type;
 
  // construct/copy/destroy
  template <class U, class = decltype(
    detail::FUN<T>(std::declval<U>()),
    std::enable_if_t<!std::is_same_v<reference_wrapper, std::remove_cvref_t<U>>>()
  )>
  constexpr reference_wrapper(U&& u) noexcept(noexcept(detail::FUN<T>(std::forward<U>(u))))
    : _ptr(std::addressof(detail::FUN<T>(std::forward<U>(u)))) {}
  reference_wrapper(const reference_wrapper&) noexcept = default;
 
  // assignment
  reference_wrapper& operator=(const reference_wrapper& x) noexcept = default;
 
  // access
  constexpr operator T& () const noexcept { return *_ptr; }
  constexpr T& get() const noexcept { return *_ptr; }
 
  template< class... ArgTypes >
  constexpr std::invoke_result_t<T&, ArgTypes...>
    operator() ( ArgTypes&&... args ) const {
    return std::invoke(get(), std::forward<ArgTypes>(args)...);
  }
 
private:
  T* _ptr;
};

虽然std::reference_wrapper的实现已经在这里和这里讨论过了,但没有一个讨论我感到困惑的构造函数实现。我的困惑是:1.(构造函数是一个模板函数,采用不同于模板类param T的类型param(U(。我见过一个类的成员函数是模板函数,取决于不同的类型params,然后是模板类的类型param,但我想不出它在这里是如何工作的。这里有一个相关的问题,但我无法将其与我的困惑联系起来。2.(我看到构造函数中的第二个类型参数被进一步用于sfinae输出一些东西,但我不明白detail::FUN<T>(std::declval<U>())是如何计算的。

有人能解释一下吗?

编辑:这是从microsoft.docs添加的示例例如:

    int i = 1;
    std::reference_wrapper<int> rwi(i);  // A.1
    rwi.get() = -1;
    std::cout << "i = " << i << std::endl; //Prints -1

随着reference_wrapper的实现,从A.1开始,reference_wraper的构造函数是如何调用的?假设detail::FUN<T>(std::declval<U>()将用detail::FUN<T>(std::declval<int>()调用,由于deleted过载,应该是替换失败(假设std::declval<int>将被读取为对int的右值引用(。我在这里错过了什么?

当您想要;转发参考";,在这种情况下,U&&,但同时限制可以绑定到它的内容

T的推导有以下推导指南的帮助。当U被推断为右值引用时,detail::FUN<T>(std::declval<U>())通过选择已删除的重载并生成无效表达式来确保构造函数被禁用。如果U是另一个引用包装,在这种情况下应该选择复制构造函数,那么它也会被禁用。

以下是有效和无效reference_wrapper的几个例子:

int i = 1; 
// OK: FUN<int>(std::declval<int&>()) is valid
std::reference_wrapper<int> rwi(i); 
// error: forming pointer to reference
std::reference_wrapper<int&> rwi2(i); 
// OK, uses deduction guide to find T = int
std::reference_wrapper rwi3(i);
std::reference_wrapper rwi4(++i);
// error: cannot deduce T, since there is no deduction guide for
// rvalue reference to T
std::reference_wrapper rwi5(std::move(i));
// error: substitution failure of FUN<int>(int&&)
std::reference_wrapper<int> rwi6(std::move(i));
std::reference_wrapper<int> rwi7(i++);
std::reference_wrapper<int> rwi8(i + i);
std::reference_wrapper<int> rwi9(2);

正如您所看到的,对已删除的FUN<T>(T&&)的调用只在最后4种情况下起作用:当您显式指定T,但试图从右值构造时。

相关内容

  • 没有找到相关文章

最新更新