1 linux编译C++代码出现error: use of deleted function std::atomic::atomic(const std::atomic&)

今天在Ubuntu,GCC版本为7.5编译类似以下C++代码时出现了error: use of deleted function ‘std::atomic<bool>::atomic(const std::atomic<bool>&)错误。

#include <iostream>
#include <atomic>

class Test
{
public:
    Test()
    {

    }

    virtual~Test()
    {

    }
private:
    std::atomic<bool> m_IsInit = false;
};

int main()
{
    Test test;

    return 0;
}

首先这段代码在Visual Studio 2019上编译是没有错误的,但是在GCC7.5上出现了错误。

这个错误的字面意思是使用了删除的函数,原因是原子变量无法使用拷贝构造函数。上面出问题的代码

std::atomic<bool> m_IsInit = false;

主要是给原子变量初始化,这个代码等价于

std::atomic<bool> m_IsInit = std::atomic<bool>(false);

上述代码会先生成一个临时的std::atomic<bool>对象,然后再用这个对象给m_IsInit进行拷贝构造,隐式调用了拷贝构造函数,而原子变量是不能进行拷贝构造的,这在官方文档中也进行了说明,

C++ – linux编译C++代码出现error: use of deleted function std::atomic<bool>::atomic(const std::atomic<bool>&)-StubbornHuang Blog

相应链接:https://en.cppreference.com/w/cpp/atomic/atomic/operator%3D

这个限制只是在原子变量初始化时生效,初始之后时可以使用赋值操作符的,所以有多种解决方案去处理这个错误。

2 解决方案

2.1 在构造函数中赋值

将原子变量的赋值从初始化列表写到构造函数中即可,修改后的代码如下:

#include <iostream>
#include <atomic>

class Test
{
public:
    Test()
    {
        m_IsInit = false;
    }

    virtual~Test()
    {

    }
private:
    std::atomic<bool> m_IsInit;
};

int main()
{
    Test test;

    return 0;
}

2.2 直接初始化或使用初始化列表初始化

将代码行

std::atomic<bool> m_IsInit = false;

修改为

std::atomic<bool> m_IsInit{false};     // 方法1
std::atomic<bool> m_IsInit= {false};  // 方法2