ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

c – 错误C2783:’_Ty \u0026\u0026 std :: forward(remove_reference :: type \u0026\u0026)throw()’:无法推

2019-07-28 09:10:53  阅读:178  来源: 互联网

标签:perfect-forwarding forwarding-reference c c11


我有一个带有push函数的并发队列的模板化实现,如下所示:

template <typename T>
class concurrent_queue
{
public:

    // other code...

    void push(const T& item)
    {
        std::unique_lock<std::mutex> mlock(mutex);
        queue.push_back(std::forward(item));
        mlock.unlock();
        notEmpty.notify_one();
    }

private:

    std::deque<T>               queue;
    std::mutex                  mutex;
    // other stuff...
};

稍后,我实例化它并像这样使用它:

concurrent_queue<c2Type> m_queue;  // c2 type is some struct declared previously

然后我尝试推送队列上的项目,我得到上面提到的编译器错误:

c2Type c2message;

// fill in the message struct...
m_queue.push(c2message);

我之前成功地使用了队列作为线程池实现的一部分,它存储了std :: function对象.我不明白为什么在这种情况下不能推断出类型.有什么想法吗?

解决方法:

像“左值”和“右值”这样的值类别是表达式的属性.名称变量的表达式总是lvalue表达式,即使它们命名的变量具有对some_type的类型rvalue引用.

我们使用lvalue-reference和rvalue-references来绑定不同类别的表达式:按照惯例,我们将lvalue-references视为绑定到lvalues,并将rvalue-references视为绑定到rvalues.

std :: forward旨在恢复我们假设引用所引用的值类别.例如:

int   i = 42;
int&  l = i;
int&& r = 21;

l // this expression is an lvalue-expression
r // this expression is an lvalue-expression, too (!)

std::forward<int& >(l) // this function-call expression is an lvalue-expression
std::forward<int&&>(r) // this function-call expression is an rvalue-expression

std :: forward,是“普通函数”,不能仅仅通过使用参数来恢复值类别.两个参数都是左值表达式.您必须通过手动提供template-argument来指定要还原的值类别.

只有当我们有一个我们不知道先验的参考是否是右值参考或左值参考时,这才有意义.编写使用带转发引用的完美转发的函数时就是这种情况.

顺便说一句,我们想要恢复值类别,以允许另一个函数从我们收到的参数移开.如果我们收到一个rvalue参数,我们想传递一个rvalue,以允许被调用的函数移动.

对于像OP中那样的函数:

void push(const T& item)

我们知道item有对const T的类型lvalue引用.因此,我们不需要std :: forward:

void push(const T& item) {
    // ...
    queue.push_back(item); // pass the lvalue argument as an lvalue
    // ...
}

如果我们添加另一个重载:

void push(T&& item)

我们仍然不需要std :: forward,因为该参数项的类型总是rvalue-reference to T(假设T不是引用类型):

void push(T&& item) {
    // ...
    queue.push_back(std::move(item)); // pass the rvalue argument as an rvalue
    // ...
}

只有我们有类似的东西

template<typename U>
void push(forwarding_reference<U> item)

其中forwarding_reference< U>可以是左值引用或右值引用,那么我们需要std :: forward:

template<typename U>
void push(forwarding_reference<U> item) // not C++, read on
{
    // ...
    queue.push_back(std::forward<U>(item)); // pass lvalue arguments as lvalues
                                            // and rvalue arguments as rvalues
    // ...
}

由于实施细节,我们必须将上述内容写成:

template<typename U>
void push(U&& item) {
    // ...
    queue.push_back(std::forward<U>(item)); // pass lvalue arguments as lvalues
                                            // and rvalue arguments as rvalues
    // ...
}

注意上面的U&& item不是rvalue-reference,而是转发引用.要获得转发参考,您需要一个带有一些模板类型参数X的函数模板和一个X&&amp ;;形式的函数参数. X.

标签:perfect-forwarding,forwarding-reference,c,c11
来源: https://codeday.me/bug/20190728/1560270.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有