QT信号量QSemaphore基础学习

82 篇文章 1 订阅

QT信号量QSemaphore基础学习

QSemaphore作为QT中的信号量,相当于多把互斥锁,QMutex只锁一次,而QSemaphore能锁多次,且控制多个条件。
如下:先创建一个客户类和一个线程类来使用信号量,
QSemaphore中主要函数为:
acquire(int n = 1)获取资源,默认获取第一个资源,相当于获取资源后加锁,如果没有资源会阻塞到当前位置;
available()获取当前可用资源数量;
release(int n = 1)释放n个资源,相当于开锁;
tryAcquire(int n = 1)尝试获取第n个资源,相当于开启n把锁,开锁成功后返回true,失败返回false,不阻塞。
头文件:

#include <QMainWindow>
#include <QThread>
#include <QSemaphore>
class mythread;

namespace Ui {
class mysemaphore;
}

class mysemaphore : public QMainWindow
{
    Q_OBJECT

public:
    explicit mysemaphore(QWidget *parent = 0);
    ~mysemaphore();

public slots:
    void deleteSlot();
private slots:
    void on_pushButton_clicked();

private:
    Ui::mysemaphore *ui;
    mythread* m_thread;
};


class mythread : public QThread
{
    Q_OBJECT

public:
    mythread();

    virtual void run();
public slots:
    void useSlot();

public:
    QSemaphore m_semaphore;
    int index;
};

cpp文件:

#include "mysemaphore.h"
#include "ui_mysemaphore.h"
#include <QDebug>
mysemaphore::mysemaphore(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::mysemaphore)
{
    ui->setupUi(this);
    m_thread = new mythread();
    connect(ui->threadbtn,SIGNAL(clicked()),m_thread,SLOT(useSlot()));
    connect(m_thread,SIGNAL(finished()),this,SLOT(deleteSlot()));
}

mysemaphore::~mysemaphore()
{
    delete ui;
}

void mysemaphore::on_pushButton_clicked()
{
    m_thread->start();
}


void mysemaphore::deleteSlot()
{
    m_thread->wait();
    m_thread->deleteLater();
}

void mysemaphore::on_lineEdit_textChanged(const QString &arg1)
{
    m_thread->index = ui->lineEdit->text().toInt();
}


mythread::mythread() : m_semaphore(0),index(0)
{

}

void mythread::run()
{
    while(1)
    {
        //获取可用资源数量
        switch (index)
        {
        case 0:
            //获取一个资源
            m_semaphore.acquire();
            qDebug()<<"1--unlock--sem"<<m_semaphore.available()<<"---type=0";
            break;
        case 1:
            //获取2个资源
            m_semaphore.acquire(2);
            qDebug()<<"2--unlock--sem"<<m_semaphore.available()<<"---type=1";
            break;
        case 2:
            // 尝试获取资源
            if (m_semaphore.tryAcquire())
            {
                qDebug()<<"try--unlock--sem"<<m_semaphore.available()<<"---type=2";
            }
            break;
        case 3:
            // 尝试获取2个资源
            if (m_semaphore.tryAcquire(2))
            {
                qDebug()<<"try--unlock-2-sem"<<m_semaphore.available()<<"---type=3";
            }
            break;
        case 4:
            // 控制多个条件
            m_semaphore.acquire();
            qDebug()<<"--unlock--1"<<m_semaphore.available()<<"---type=4";
            m_semaphore.release();//释放1个资源
            m_semaphore.acquire(2);
            qDebug()<<"--unlock--2"<<m_semaphore.available()<<"---type=4";
            m_semaphore.release(2);//释放2个资源
            m_semaphore.acquire(3);
            qDebug()<<"--unlock--3"<<m_semaphore.available()<<"---type=4";
            break;
        default:
            qDebug()<<"no exsit"<<m_semaphore.available();
            break;
        }
    }
}

void mythread::useSlot()
{
    qDebug()<<"click--";
    // 释放一个资源
    m_semaphore.release();
}

结果如下:
在这里插入图片描述

  • 1
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Qt提供了QSemaphore类来实现信号量功能,可以用于实现一个生产者多个消费者的场景。 首先,我们需要创建一个QSemaphore对象,作为生产者和消费者之间共享的信号量。在这个例子中,我们假设生产者线程负责生产产品,消费者线程负责消费产品。 生产者的逻辑如下: 1. 首先,获取信号量的锁定,如果信号量的计数器为0,则阻塞等待信号量的释放; 2. 当信号量的计数器不为0时,生产者线程开始生产产品,并进行相应的操作; 3. 完成产品的生产后,释放信号量的锁定,并增加信号量的计数器。 消费者的逻辑如下: 1. 首先,获取信号量的锁定,如果信号量的计数器为0,则阻塞等待信号量的释放; 2. 当信号量的计数器不为0时,消费者线程开始消费产品,并进行相应的操作; 3. 完成产品的消费后,释放信号量的锁定,并增加信号量的计数器。 下面是一个简化的示例代码: ```cpp QSemaphore semaphore; // 创建信号量对象 QVector<QString> products; // 存放产品的容器 // 生产者线程函数 void producer() { while (true) { semaphore.acquire(); // 获取信号量的锁定 // 生产产品的操作 QString product = generateProduct(); products.append(product); semaphore.release(); // 释放信号量的锁定 QThread::sleep(1); // 等待一段时间 } } // 消费者线程函数 void consumer() { while (true) { semaphore.acquire(); // 获取信号量的锁定 // 消费产品的操作 if (!products.isEmpty()) { QString product = products.takeFirst(); consumeProduct(product); } semaphore.release(); // 释放信号量的锁定 QThread::sleep(1); // 等待一段时间 } } int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); // 创建生产者和消费者线程 QThread producerThread; QThread consumerThread1; QThread consumerThread2; QThread consumerThread3; // 信号量关联线程 semaphore.setInitialValue(1); semaphore.moveToThread(&producerThread); semaphore.moveToThread(&consumerThread1); semaphore.moveToThread(&consumerThread2); semaphore.moveToThread(&consumerThread3); // 启动线程 producerThread.start(); consumerThread1.start(); consumerThread2.start(); consumerThread3.start(); return a.exec(); } ``` 在上面的示例代码中,我们创建了一个QSemaphore对象作为信号量,并使用`acquire()`和`release()`函数来获取和释放信号量的锁定。生产者线程通过`acquire()`函数获取信号量的锁定,如果信号量的计数器为0,则线程会阻塞等待信号量的释放。消费者线程也使用相同的方法来获取和释放信号量的锁定。 这种方式实现了一个生产者多个消费者的场景,多个消费者线程可以在信号量的控制下轮流进行产品的消费。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

东方忘忧

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值