【问题标题】:TBB compiler error - 'my_task': references must be initializedTBB 编译器错误 - “my_task”:必须初始化引用
【发布时间】:2018-05-10 14:47:38
【问题描述】:

我在项目的多个地方使用 TBB。但似乎自从我将 Visual Studio 从 15.6.X(X 是最新版本)更新到 15.7.1 后,我在几个地方遇到了编译器错误,告诉我

[...]tbb\task_group.h(94): 错误 C2530: 'my_task': 必须初始化引用

查看引用的代码(tbb/task_group.h):

//! Base class for types that should not be assigned.
class no_assign {
    // Deny assignment
    void operator=( const no_assign& );
public:

#if __GNUC__
    //! Explicitly define default construction, because otherwise gcc issues gratuitous warning.
    no_assign() {}
#endif /* __GNUC__ */
};

//! Base class for types that should not be copied or assigned.
class no_copy: no_assign {
    //! Deny copy construction
    no_copy( const no_copy& );
public:
    //! Allow default construction
    no_copy() {}
};

// ...

class ref_count_guard : internal::no_copy {
    task& my_task;  // compiler error occurs here
public:
    ref_count_guard( task& t ) : my_task(t) {
        my_task.increment_ref_count();
    }
    ~ref_count_guard() {
        my_task.decrement_ref_count();
    }
};

我不明白为什么编译器在那里抱怨,因为引用是由构造函数初始化的。在我的代码中找到问题也不是那么容易,因为编译器错误发生在每个使用 TBB 的源文件中,而且我认为自上次成功编译以来我没有更改任何内容(除了更新 VS)。

我想到的一种可能性与this question 有关。如果默认情况下 msvc 以某种方式继承了基类构造函数,则会继承默认构造函数来解释错误。但是testing 这种情况似乎反驳了它(代码编译时)。

为什么 msvc 在这里抱怨?

更新

这个最小的例子重现了我系统上的错误:

#include <vector>
#include <tbb/tbb.h>
#include <tbb/flow_graph.h>       

void main()
{
    std::vector<int> src{1, 2, 3, 4, 5};
    tbb::parallel_for_each(src.begin(), src.end(), [](int) { });
}

更新 2

看起来只是包含tbb/tbb.h 会导致错误发生。我什至不需要打电话。用新的编译器版本重建 tbb 也没有帮助。

编辑

github 上的交叉问题。

【问题讨论】:

  • "// Deny assignment void operator=( const no_assign&amp; );" 由void operator(const no_assign&amp;) = delete; 更好/更明确地完成,您甚至根本不需要no_assign 类。只是= delete 类中不应赋值的赋值运算符,而不是从no_assign 继承。更少的代码,更明确的;赢了,赢了。
  • @JesperJuhl 那是 TBB 内部代码。那不是我写的。我知道= delete 是更好的选择...
  • "` 显式定义默认构造` ..." - 只需使用= default 那么默认构造函数也将是微不足道的,而不是用户定义的(如果可能的话,这是可取的)。
  • @JesperJuhl 再次强调:上面显示的代码不是我的代码。它来自 TBB 头文件。
  • @ntfs.hard 是的,这个问题是我打开的。我在此处和 github 上添加了链接。

标签: c++ visual-studio visual-c++ c++17 tbb


【解决方案1】:

只需删除 /permissive-(例如,在 C/C++ 选项中将 Comformance Mode 设置为 No)即可解决此问题。我想英特尔很快就会解决这个问题。

【讨论】:

  • 感谢您的观察,它允许重现该问题。
【解决方案2】:

这是使用/permissive- 选项时的编译器错误。可以用以下代码复现:

struct U {
    template<typename T>
    void foo() {
        class A {
            int& iref;
        public:
            A(int& ir) : iref(ir) { }
        };
        int the_answer = 42;
        A a(the_answer);
    }
};

int main() {
    U u;
    u.foo<int>();
    return 0;
}

代码完全有效,兼容 C++。如您所见,引用在构造函数的成员初始化器列表中显式初始化。此外,这似乎是 VS 中的回归,因为至少 VS 2017 的初始版本(15.0.something)使用/permissive- 编译此代码。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-09-04
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-12-31
    • 1970-01-01
    相关资源
    最近更新 更多