3

我有这个简单的课程:

struct Worker
{
        Worker() : done{false} {}
        Worker(const Worker& rhs) : done{rhs.done}, qworker{} {}
        Worker(Worker &&rhs) : done{rhs.done}
        {
            qworker = std::move(rhs.qworker);
        }
...
}

这可以用 gcc-4.7.2 编译,但是如果我尝试使用这个版本,我会得到一个错误

struct Worker
{
        Worker() : done{false} {}
        Worker(const Worker& rhs) : done{rhs.done}, qworker{} {}
        Worker(Worker &&rhs) : done{rhs.done}
                             , qworker{std::move(rhs.qworker)} // <- ERROR
        {
        }
...
}

为什么?

In file included from tlog.cpp:8:0:
log11.hpp: In member function ‘void Log11::Worker::run()’:
log11.hpp:34:29: error: ‘class std::vector<std::function<void()> >’ has no member named ‘pop_front’
In file included from /usr/include/c++/4.7/thread:39:0,
                 from tlog.cpp:3:
/usr/include/c++/4.7/functional: In instantiation of ‘static void std::_Function_handler<void(_ArgTypes ...), _Functor>::_M_invoke(const std::_Any_data&, _ArgTypes ...) [with _Functor = std::vector<std::function<void()> >; _ArgTypes = {}]’:
/usr/include/c++/4.7/functional:2298:6:   required from ‘std::function<_Res(_ArgTypes ...)>::function(_Functor, typename std::enable_if<(! std::is_integral<_Functor>::value), std::function<_Res(_ArgTypes ...)>::_Useless>::type) [with _Functor = std::vector<std::function<void()> >; _Res = void; _ArgTypes = {}; typename std::enable_if<(! std::is_integral<_Functor>::value), std::function<_Res(_ArgTypes ...)>::_Useless>::type = std::function<void()>::_Useless]’
log11.hpp:20:78:   required from here
/usr/include/c++/4.7/functional:1926:2: error: no match for call to ‘(std::vector<std::function<void()> >) ()’
4

1 回答 1

3

根据 C++11 标准std::function,有一个不受约束的构造函数模板,可以接受任何参数类型:

template<class F> function(F f);

当你说qworker{std::move(rhs.qworker)}这首先尝试调用构造函数时std::initializer_list<std::function<void()>>。由于上面显示的不受约束的构造函数模板,astd::function<void()>可以从任何类型构造,因此您会得到一个initializer_list带有一个成员的 a,如下所示:

{ std::function<void()>{std::move(rhs.qworker)} }

这是无效的,因为rhs.qworker它不是可调用对象,但仅当您尝试调用函数对象时才会发生错误。

如果您说qworker(std::move(rhs.qworker)),那么初始化列表构造函数不是候选者,而是调用移动构造函数。

有一个针对标准 ( LWG 2132 ) 的缺陷报告,该标准通过防止function(F)调用构造函数模板来解决此问题,除非参数是可调用对象。这可以防止initializer_list<function<void()>>创建,而是qworker{std::move(rhs.qworker)}按预期调用移动构造函数。GCC 4.7 没有实现 LWG 2132 的解决方案,但 GCC 4.8 可以。

于 2013-04-07T17:13:31.613 回答