繁体   English   中英

您是否曾将 C++ RValue 引用参数标记为 const

[英]Would you ever mark a C++ RValue reference parameter as const

我一直在切换模板工厂函数以使用(并理解)std::forward 来支持右值和移动语义。 我通常用于模板类的样板工厂函数总是将参数标记为 const:

#include <iostream>
#include <utility>

template<typename T, typename U>
struct MyPair{
    MyPair(const T& t, const U& u):t(t),u(u){};

    T t;
    U u;
};

template<typename T, typename U>
std::ostream& operator<<(std::ostream& os, const MyPair<T,U>& pair){
    os << "(" << pair.t << ")=>" << pair.u;
    return os;
}

template<typename T, typename U>
MyPair<T,U> MakeMyPair(const T& t, const U& u){
    return MyPair<T,U>(t,u);
}

using namespace std;
int main(int argc, char *argv[]) {    

    auto no_forward = MakeMyPair(num, num);
    std::cout << no_forward << std::endl;

    auto no_forward2 = MakeMyPair(100, false);
    std::cout << no_forward2 << std::endl;
}

按预期编译。 最初我将 MakeMyPair 转换为也将参数作为 const 传递,但这不会在我的 Mac 上使用 XCode 4.6 编译:

//$ clang --version
//Apple LLVM version 4.2 (clang-425.0.24) (based on LLVM 3.2svn)
//Target: x86_64-apple-darwin12.2.0
//Thread model: posix


template<typename T, typename U>
MyPair<T,U> MakeMyPair_Forward(const T&& t, const U&& u){
    return MyPair<T,U>(std::forward<const T>(t),std::forward<const U>(u));
}

int main(int argc, char *argv[]) { 
    int num = 37;
    auto anotherPair = MakeMyPair_Forward(num, true); //This won't work

    auto allRvalues = MakeMyPair_Forward(73, false);   //will compile 
    std::cout << allRvalues  << std::endl;
}

没有用于调用“MakeMyPair_Forward”候选函数的匹配函数 [with T = int, U = bool] 不可行:第一个参数没有从“int”到“const int &&”的已知转换

这从http://en.cppreference.com/w/cpp/utility/forward中是有道理的,它指出 const 是推导出来的,我正在传递左值。

  • 如果对 wrapper() 的调用传递右值 std::string,则 T 被推导为 std::string(不是 std::string&、const std::string& 或 std::string&&),并且 std::forward 确保将右值引用传递给 foo。
  • 如果对 wrapper() 的调用传递了一个 const 左值 std::string,则 T 被推导为 const std::string&,并且 std::forward 确保将一个 const 左值引用传递给 foo。
  • 如果对 wrapper() 的调用传递了一个非常量左值 std::string,则 T 被推导为 std::string&,并且 std::forward 确保将非常量左值引用传递给 foo。

删除 const 可以像我想要的那样使用右值和左值。 只有将右值作为类型传递才能在 MakeMyPair_Forward 的参数上使用 const。

//This works for rvalues and lvalues
template<typename T, typename U>
MyPair<T,U> MakeMyPair_Forward(T&& t, U&& u){
    return MyPair<T,U>(std::forward<const T>(t),std::forward<const U>(u));
}

所以,问题。 作为参数传递时,将右值引用标记为 const 是否有意义? 我不能改变右值,这只是暂时的。 在完成并修复我的代码后,我有点惊讶它用 const 编译。 为什么要将右值参数标记为 const? 重点是只提供一个接受右值的 API 吗? 如果是这样,您会不会使用类型特征来防止左值引用? https://stackoverflow.com/a/7863645/620304

谢谢。

所以,问题。 作为参数传递时,将右值引用标记为 const 是否有意义?

这是在 C++11 标准中完成的一个地方:

template <class T> reference_wrapper<T> ref(T&) noexcept;
template <class T> reference_wrapper<const T> cref(const T&) noexcept;
template <class T> void ref(const T&&) = delete;
template <class T> void cref(const T&&) = delete;

const T&&用于捕获所有右值,无论是否为 const,并将它们扔给编译时错误,同时允许左值,甚至是const左值绑定和工作。

现在这也可以通过T&&enable_if约束来完成。 但是,如果 C++ 在过去几十年中教会了我们一件事:不要在语言设计中烧掉任何桥梁。 C++ 程序员经常会找到一种聪明的方法来使用最初被认为无用的语言功能。 正是本着这种精神, const T&&被保留为合法选项。

另一个可能的用例,不是被语言采用,但可能是,是来自原始指针的智能指针构造函数,例如:

// this is NOT the actual ctor in the language but could have been
// see explanation in above link
unique_ptr(T* const&& p) : ptr{p} {}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM