Qt中线程同步的几种方法详解

1、QMutex类

       QMutex类就像一把锁,在互斥量以前上锁(QMutex::lock()),而后在使用完互斥量以后解锁(QMutex::unlock())。好比下面的代码:函数

void someMethod()
{
    mutex.lock();
    qDebug()<<"Hello";
    qDebug()<<"World";
    mutex.unlock();
}
 
class Thread1 : public QThread  
{
protected:
    virtual void run()
    {
        someMethod();
    }
};
 
class Thread2 : public QThread  
{
protected:
    virtual void run()
    {
        someMethod();
    }
};

如上面的代码,在函数someMethod里面有两条语句,若是有两个线程启动以后,这两个线程都将调用这个函数(run函数即为线程启动后自动执行的函数),则可能会出现的结果是Hello Hello World World。但显然,这并非咱们想要的,咱们但愿的是每一个线程能够一次性执行完someMethod函数里面的代码。这个时候咱们即可以在函数里面给函数体加上锁,而后在结束的时候解锁。spa

这里须要注意的是,若是一个线程试图向一个已经被其它线程上锁了互斥量上锁的话,这个线程将被阻塞(挂起),直到这个互斥量被解锁。若是一个线程但愿本身在试图对一个上锁了的互斥量进行访问的时候可以不被阻塞(而是当即返回),能够将lock()函数替换为tryLock()函数,这个函数的效果是:若是线程正在试图访问的互斥量已经被上锁了,那么能够当即返回而不被阻塞。.net

2、QMutexLocker便利类

使用 QMutex 对互斥量进行加锁解锁比较繁琐,在一些复杂的函数或者抛出C++异常的函数中都很是容易发生错误。可使用一个方便的 QMutexLocker 类来简化对互斥量的处理。首先,QMutexLocker类的构造函数接收一个QMutex对象做为参数而且上锁,而后在析构函数中自动对其进行解锁。以下代码:线程

QMutex mutex;
 
void someMethod()
{
    QMutexLocker locker(&mutex);
    qDebug()<<"Hello";
    qDebug()<<"World";
}

这里建立一个QMutexLocker类实例,在这个实例的构造函数中将对mutex对象进行加锁。而后在析构函数中自动对mutex进行解锁。解锁的工做不须要显示地调用unlock函数,而是根据QMutexLocker对象的做用域绑定在一块儿了。指针

3、QReadWriteLock类

      前两种保护互斥量的方法比较绝对,其达到的效果是:无论我要对互斥量作些是什么,我都要一我的霸占着,即便我只是看看它,也不能让别人看。这会使得这个互斥量资源的使用率大大降低,形成资源等待等问题。code

      因而,咱们能够对线程对互斥量的操做进行分类:读和写。有几种状况:对象

一、若是我只是看看的话,你也能够看,你们看到的都是正确的结果;blog

二、若是我要看这个数据,你是不能改的,否则我看到的结果就不知道是什么了;资源

三、我在改的时候,你不能看,不然我可能会让你看到不正确的结果;作用域

四、我在改的时候,你固然不能改了。

所以,咱们能够对QMutex锁进行升级,将其升级为QReadWriteLock,QMutex加锁的方法是lock(),而QReadWriteLock锁有两种锁法:设置为读锁(lockForRead())和写锁(lockForWrite())。代码以下:

QReadWriteLock lock;
void someMethod()
{
lock.lockForRead(); //为读而锁
//lock.lockForWrite(); //为写而锁
qDebug()<<"Hello";
qDebug()<<"World";

lock.unlock();  //解锁
}

因而可能有如下四种状况:

一、一个线程试图对一个加了读锁的互斥量进行上读锁,容许;

二、一个线程试图对一个加了读锁的互斥量进行上写锁,阻塞;

三、一个线程试图对一个加了写锁的互斥量进行上读锁,阻塞;

四、一个线程试图对一个加了写锁的互斥量进行上写锁,阻塞。

因此能够看出,读写锁比较适用的状况是:须要屡次对共享的数据进行读操做的阅读线程。

4、QReadLocker便利类和QWriteLocker便利类对QReadWriteLock进行加解锁

和QMutex与QMutexLocker类的关系相似,关于读写锁也有两个便利类,读锁和写锁,QReadLocker和QWriteLocker。它们的构造函数都是一个QReadWriteLock对象,不一样的是,在QReadLocker的构造函数里面是对读写锁进行lockForRead()加锁操做,而在QWriteLocker的构造函数里面是对读写锁进行lockForWrite()加锁操做。而后解锁操做unlock()都是在析构函数中完成的。

void write()
{
QReadLocker locker(&lock);
..........
}

void read()
{
QWriteLocker locker(&lock);
..............
}

QReadLocker和QWriteLocker的成员函数都如出一辙,退出函数的时局部变量locker会自动销毁,讲lock自动解锁。也能够调用locker.unlock()给lock解锁,而后再调用locker.relock()再锁住lock。也能够调用locker.readWriteLock()获取创建locker时引入的那个lock的指针。

5、信号量QSemaphore

前面的几种锁都是用来保护只有一个变量的互斥量的。可是还有些互斥量(资源)的数量并不止一个,好比一个电脑安装了2个打印机,我已经申请了一个,可是我不能霸占这两个,你来访问的时候若是发现还有空闲的仍然能够申请到的。因而这个互斥量能够分为两部分,已使用和未使用。一个线程在申请的时候,会对未使用到的部分进行加锁操做,若是加锁失败则阻塞,若是加锁成功,即又有一个资源被使用了,因而则将已使用到的部分解锁一个。

以著名的生产者消费者问题为例,分析问题:生产者须要的是空闲位置存放产品,结果是可取的产品多了一个。因而,咱们能够定义两个信号量:QSemaphore freeSpace和QSemaphore usedSpace,前者是给生产者使用的,后者是给消费者使用的。

6、条件触发QWaitCondition

QWaitCondition最大的好处,我以为,是能在一个线程中唤醒一个或多个其它线程,固然前提是,其它线程在等待某个QWaitCondition,不然不起做用,你唤醒也没用。QWaitCondition必须与QMutex或者QReadwriteLock一块儿用。

  • 1
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
QT,有几种方法可以新建线程。一种方法是继承QThread类,然后重写run方法。在这种方法,除了run方法之外的所有操作都是在主线程执行的。另一种方法是使用moveToThread将新建线程转移到继承QObject的新类实例。这种方法,所有的成员函数都是在线程执行的。还有一种方法是继承QRunnable类,重新实现run函数,并结合QThreadPool实现线程池。在这种方法,可以通过构建一个本地的线程池,并设置最大线程数来控制线程的执行。\[1\]\[2\] 另外,还有两种方法是继承QThread类并重写run函数,以及继承QObject类并调用moveToThread函数。这两种方法不需要继承类,可以直接将需要执行的函数放到线程运行。在第一种方法,可以通过调用start函数启动线程,并自动调用run函数。可以使用isRunning函数判断线程是否已启动,使用terminate函数终止线程,使用wait函数等待线程终止。在第二种方法,可以通过调用moveToThread函数将对象移动到新线程执行。\[3\] #### 引用[.reference_title] - *1* [QT新建线程的几种方法](https://blog.csdn.net/li3781695/article/details/88233286)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Qt开启线程的五种方法](https://blog.csdn.net/m0_73443478/article/details/129556890)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值