我一直试图从这里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.) 构造函数是一个模板函数,采用U
与模板类 param 不同的类型 param() T
。我已经看到一个类的成员函数是模板函数,并且取决于不同的类型参数,然后是模板类的类型参数,但我想不出它在这里是如何工作的。这里有一个相关的问题,但我无法将它与我的困惑联系起来。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_wrapper的构造函数呢?假设detail::FUN<T>(std::declval<U>()
将调用detail::FUN<T>(std::declval<int>()
, 应该是由于deleted
重载而导致的替换失败(假设std::declval<int>
将被读取为对 int 的右值引用)。我在这里想念什么?