我有一个带有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
个对象。我不明白为什么在这种情况下不能推断出类型。有什么想法吗?
答案 0 :(得分:7)
价值类别如&#34;左值&#34;和&#34; rvalue&#34;是表达式的属性。名称变量的表达式始终是左值表达式,即使它们命名的变量具有右值引用some_type
的类型。
我们使用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
,&#34;普通函数&#34;,仅仅通过使用参数无法恢复值类别。两个参数都是左值表达式。您必须通过手动提供template-argument来指定要还原的值类别。
如果我们有一个参考,我们不知道它是左值参考还是左值参考,这只是有意义的。编写使用完美转发与转发引用的函数时就是这种情况。
顺便说一句,我们想要恢复值类别以允许另一个函数从我们收到的参数移开。如果我们收到一个rvalue参数,我们想传递一个rvalue,以允许被调用的函数移动。
对于像OP中那样的函数:
void push(const T& item)
我们知道item
具有左值引用const T
的类型。因此,我们不需要std::forward
:
void push(const T& item) {
// ...
queue.push_back(item); // pass the lvalue argument as an lvalue
// ...
}
如果我们添加另一个重载:
void push(T&& item)
我们仍然不需要std::forward
,因为该参数的类型item
始终是 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&& x
形式的函数参数。