【发布时间】:2012-04-17 14:35:38
【问题描述】:
以下程序未在 VS11 beta、gcc 4.5 或 clang 3.1 中构建
#include <thread>
#include <memory>
int main() {
std::unique_ptr<int> p;
std::thread th([](std::unique_ptr<int>) {
},std::move(p));
th.join();
}
这是因为参数类型不可复制,但实现会尝试复制它。
据我所知,这个程序格式正确,应该可以运行。 std::thread 的要求似乎意味着可移动的、不可复制的参数应该在这里工作。具体来说,它表示可调用对象和每个参数都应满足 MoveConstructible 的要求,并且INVOKE(DECAY_COPY(std::forward<F>(f)),DECAY_COPY(std::forward<Args>(args))...) 应是一个有效的表达式。
在这种情况下,我认为表达式类似于:
template <class T> typename std::decay<T>::type decay_copy(T&& v)
{ return std::forward<T>(v); }
std::unique_ptr<int> p;
auto f = [](std::unique_ptr<int>) {};
decay_copy(f)(decay_copy(std::move(p)));
而且我认为这不应该涉及p 的副本。 gcc 至少可以编译这个表达式,虽然 VS11 不能。
- 我对要求和参数必须是可复制的有误吗?
- 标准是否在此问题上为实现复制参数留有余地?
- 还是我尝试的实现不合格?
【问题讨论】:
-
您似乎正在通过副本传递线程参数(根据匿名函数签名)。参数类型不应该是
std::unique_ptr<int>&&或const std::unique_ptr<int>&吗? -
@André :不存在通过副本传递的东西;通过 value 传递参数将根据调用者传递左值还是右值来复制或移动。
-
@ildjarn:对不起,我的意思是“按价值”,而不是“按副本”。我突然想到按值传递参数将选择移动构造函数(如果可用)。
标签: c++ multithreading c++11 rvalue-reference