QAtomicPointer使用的错误思路

文章讲述了在C++中使用原子指针`QAtomicPointer`处理线程安全问题时,由于没有正确管理外部共享资源,导致竞态条件引发异常的实例。

以为原子指针不会崩溃,但实际发生了,究其原因还是原子指针只能保证其自身是线程安全的,它不能保证它函数之外的代码可以线程安全。

class MyClassAtomic
{
public:
    MyClassAtomic() 
    {
        
    };
    ~MyClassAtomic() 
    {
    };

    static MyClassAtomic* instance()
    {
        static MyClassAtomic* pIns = nullptr;
        if (pIns == nullptr)
        {
            pIns = new MyClassAtomic();
        }
        return pIns;
    }

    void setStr(std::string* pstr)
    {
        auto pPtrStr = new QSharedPointer<std::string>(pstr);

        auto pOldStr = atomicPtr.fetchAndStoreRelaxed(pPtrStr);
        delete pOldStr;
    }
    QSharedPointer<std::string> getStr()
    {
        return *atomicPtr.loadRelaxed();
    }


private:
    QAtomicPointer<QSharedPointer<std::string>> atomicPtr;
    
};

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    MyClassAtomic::instance();
    std::thread t1([]() 
        {
       
### Qt 多线程最佳实践 #### 工作对象模式而非重写 `run()` 方法 对于多线程的应用程序,推荐的做法不是直接继承 `QThread` 并覆写其 `run()` 函数来启动新的计算密集型任务[^1]。相反,应当定义一个新的工作类,该类派生于 `QObject` 或其他合适的基类,并将此实例移至由 `QThread` 启动的新线程上下文中运行。 ```cpp class Worker : public QObject { Q_OBJECT public slots: void process() { // 执行耗时操作... } signals: void finished(); }; ``` 接着,在主线程或其他任何地方创建一个 `Worker` 实例,并将其迁移到目标线程: ```cpp QThread* thread = new QThread; Worker* worker = new Worker; worker->moveToThread(thread); connect(worker, SIGNAL(finished()), thread, SLOT(quit())); connect(worker, SIGNAL(finished()), worker, SLOT(deleteLater())); connect(thread, SIGNAL(started()), worker, SLOT(process())); thread->start(); // 开始执行新线程内的事件循环 ``` 这种方法使得代码更加模块化、易于维护,并且充分利用了 Qt 的信号槽机制来进行线程间的安全通信[^2]。 #### 利用原子类型保障线程安全 当涉及到共享资源的操作时,使用像 `QAtomicInt`, `QAtomicPointer` 这样的原子类型可以帮助避免竞态条件的发生而无需显式的锁定开销[^3]。这些类型的成员函数提供了基本的读取/修改/存储原语,它们可以在不引起争用的情况下完成简单的计数器更新或指针交换等动作。 例如,如果有一个整数值需要被多个线程同时增加,则可以这样做: ```cpp #include <QAtomicInteger> QAtomicInteger<int> counter{0}; // 在任意数量的工作线程中调用如下代码片段 counter.fetchAndAddRelaxed(1); ``` 上述方式确保即使有多个线程几乎同时尝试改变同一个变量的内容也不会破坏数据一致性。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值