C++ std::atomic volatile 区别与联系

会用和能讲清楚,绝对是两回事,今天又遇到小同事问这个问题;

Part1: 联系,这个有点复杂=======

volatile:

作用:

C/C++ 中的 volatile 关键字和 const 对应,用来修饰变量,作为指令关键字,确保本条指令不会因编译器的优化而省略,且要求每次直接读值。

volatile与多线程语义

对于多线程编程而言,在临界区内部,可以通过互斥锁(mutex)保证只有一个线程可以访问该临界区的内容,因此临界区内的变量不需要是 volatile 的;而在临界区外部,被多个线程访问的变量应声明为 volatile 的,这也符合了 volatile 的原意:防止编译器缓存(cache)被多个线程并发用到的变量。

不过,需要注意的是,由于 volatile 关键字的“顺序执行特性”并非会完全保证语句的顺序执行(如 volatile 变量与非 volatile 变量之间的操作;又如一些 CPU 也会对语句的执行顺序进行优化),因此导致了对 volatile 变量的操作并不是原子的,也不能用来为线程建立严格的 happens-before 关系。

例:

volatile int len = 0;

如果两个线程同时 ++;

len ++;

每个线程先取到 len ==0;

都加 1;结果可能还是 1;

 根据c++11的标准:

volatile 只能防止编译器优化变量,强制载入寄存器,而不能提供原子性;

VC++中只有早期版本给volatile 提供原子性,在最新版中,微软为了遵守标准,去掉了volatile 的原子性;

C++11特性之std::atomic::store

API

void store (T val, memory_order sync = memory_order_seq_cst) volatile noexcept;
void store (T val, memory_order sync = memory_order_seq_cst) noexcept;

线程同步,请用 atomic::store 原子;

例:

  std::atomic<bool> mThreadState;
#ifndef WORKOBJECT_H
#define WORKOBJECT_H


#include <QObject>
#include <QThread>
#include <QDebug>

#include<QMutexLocker>
#include<QMutex>
#include<QWaitCondition>
#include <QDataStream>
#include <atomic>
class WorkObject : public QObject
{

  Q_OBJECT
public:
     explicit WorkObject(QObject *parent = nullptr);

signals:

public slots:
    void doWork();

    void stopWork();
public:
    bool getThreadState();
private:
    bool exitflag;

    QWaitCondition mWaitStatusCondition;
    QMutex mWaitMutex;

    std::atomic<bool> mThreadState;


};

#endif // WORKOBJECT_H

#include "workobject.h"

WorkObject::WorkObject(QObject *parent)
    : QObject(parent)
    ,mThreadState(false)
{

}
void WorkObject::doWork()
{
    mThreadState.store(true);

    while(true)
    {
        QMutexLocker locker(&mWaitMutex);
        if(!exitflag)
        {
            break;
        }
        if(mWaitStatusCondition.wait(&mWaitMutex,1))
        {
          qDebug() << "MyThread eixt ================  :" ;
           break;
        }

        qDebug() << "MyThread thread id:" << QThread::currentThreadId();
       //QThread:: sleep(5);
        //QThread::sleep(1);
    }

     mThreadState.store(false);
    qDebug() << "doWork thread exit========:" << QThread::currentThreadId();
}

void WorkObject::stopWork()
{
    QMutexLocker locker(&mWaitMutex);
   {
      //exitflag = false;
        mWaitStatusCondition.wakeAll();

    }
}

bool WorkObject::getThreadState()
{
    return  mThreadState.load();
}

获取当前线程状态;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

恋恋西风

up up up

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值