c++ - 无法将左值绑定(bind)到 A<Cv2>&&

标签 c++ constants universal-reference

我认为通用引用 (T&&) 应该采用任何类型的引用。但以下内容不起作用。

当我尝试在我正在编写的库中保持 const-correct 时,我遇到了这个问题。我是 C++ 的新手,以前从未见过这样的东西。

测试.cpp:

enum Cv_qualifier {
    constant,
    non_const
};
template <Cv_qualifier Cv> class A;
template<>
class A<Cv_qualifier::constant> {
public:
    template<Cv_qualifier Cv2> 
    void t(const A<Cv2>&& out) {}
};

template <>
class A<Cv_qualifier::non_const> {
public:
    template<Cv_qualifier Cv2> 
    void t(const A<Cv2>&& out) {}
};

int main()
{
    A<Cv_qualifier::non_const> a;
    A<Cv_qualifier::constant> b;
    a.t(b);
}

错误(使用 g++ test.cpp -std=c++11 编译):

test.cpp: In function ‘int main()’:
test.cpp:24:10: error: cannot bind ‘A<(Cv_qualifier)0u>’ lvalue to ‘const A<(Cv_qualifier)0u>&&’
     a.t(b);
          ^
test.cpp:17:10: note:   initializing argument 1 of ‘void A<(Cv_qualifier)1u>::t(const A<Cv2>&&) [with Cv_qualifier Cv2 = (Cv_qualifier)0u]’
     void t(const A<Cv2>&& out) {}
          ^

顺便说一句,在实际程序中,类A 不拥有任何实际数据,而是包含对实际保存数据的另一个类的引用。我希望这意味着当我允许 class A 的成员函数 t 接受临时对象时,我不会不断地创建间接/复制数据。

最佳答案

通用引用,或转发引用,只有在引用折叠时才会发生。它是这样工作的:

T&& & -> T&
T& && -> T&
T&& && -> T&&

这样,当您收到 T&&在模板函数中,右值引用可以折叠为其他类型的引用,具体取决于 T 的类型.在任何其他情况下,当崩溃没有发生时,SomeType&&会入住SomeType&&并且将是一个右值引用。

话虽如此,如果你想让你的函数支持转发,你可以这样做:

template <Cv_qualifier Cv> struct A;

template<>
struct A<Cv_qualifier::constant> {
    template<typename T> 
    void t(T&& out) {}
};

template <>
struct A<Cv_qualifier::non_const> {
    template<typename T> 
    void t(T&& out) {}
};

的确,现在崩溃发生了。如果你想提取 Cv_qualifier来自 T 的值,你可以让自己成为一个类型特征:

template<typename>
struct CvValue;

template<Cv_qualifier cv>
struct CvValue<A<cv>> {
    constexpr static Cv_qualifier value = cv;
};

然后,在你的函数中t ,你可以这样做:

//                   v----- This is a good practice to apply a constraint
template<typename T, std::void_t<decltype(CvValue<std::decay_t<T>>::value)>* = 0> 
auto t(T&& out) {
    constexpr auto cv = CvValue<std::decay_t<T>>::value;

    // do whatever you want with cv
}

如果你不能使用 C++17 的 std::void_t ,你可以这样实现它:

template<typename...>
using void_t = void;

但是,如果您只想测试是否 T是一个 A<...> ,使用这个:

template<typename>
struct is_A : std::false_type {};

template<Cv_qualifier cv>
struct is_A<A<cv>> : std::true_type {};

不要忘记,将它与 std::decay_t 一起使用:

template<typename T, std::enable_if_t<std::is_A<std::decay_t<T>>::value>* = 0> 
void t(T&& out) {}

关于c++ - 无法将左值绑定(bind)到 A<Cv2>&&,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40619453/

相关文章:

c++ - cocos2d-x 每次需要时制作 Sprite 或只是改变其纹理的最佳做法是什么?

c++ - 是否可以用 C++ 对 iPhone 进行编程

c++ - 如何在文件中保存 vector<vector<vector<float>>> 并检索数据?

c++ - Eigen在ios项目中的使用

c++ - 如何解释 "const unique_ptr"

c++ - 左值参数是否更喜欢左值引用参数而不是通用引用?

c++ - 如何避免 C++ 派生类中基类成员的重复拷贝

C++ const char* 到 char*

c++ - 模板推导/重载决议有利于 T&& 而不是 const T&

c++ - 用于通用引用的左值/右值 -nes 编码