QT多线程同步和异步控制

同步和异步(并发):

**同步:**在处理公共资源中,信息只能按照规定顺序逐个执行,这种处理为同步执行,比如接收到指令后插入队列,然后逐个从队列中弹出处理。
所谓同步,就是发出一个功能调用时,在没有得到结果之前,该调用就不返回或继续执行后续操作。
简单来说,同步就是必须一件一件事做,等前一件做完了才能做下一件事。

**异步(并发):**在处理公共资源中,信息可以同时一起执行,这种处理为异步执行,比如接收到指令后,每一个指令创建一个线程执行处理。
异步与同步相对,当一个异步过程调用发出后,调用者在没有得到结果之前,就可以继续执行后续操作。当这个调用完成后,一般通过状态、通知和回调来通知调用者。对于异步调用,调用的返回并不受调用者控制。
对于通知调用者的三种方式,具体如下:

状态:即监听被调用者的状态(轮询),调用者需要每隔一定时间检查一次,效率会很低。
通知:当被调用者执行完成后,发出通知告知调用者,无需消耗太多性能。
回调:与通知类似,当被调用者执行完成后,会调用调用者提供的回调函数。

同步和异步的区别:

总结来说,同步和异步的区别:请求发出后,是否需要等待结果,才能继续执行其他操作。

通常使用的connect,实际上最后一个參数使用的是Qt::AutoConnection类型:Qt支持6种连接方式。当中3中最主要:
1.Qt::DirectConnection(直连方式)(信号与槽函数关系类似于函数调用。同步运行) 当信号发出后。对应的槽函数将马上被调用。emit语句后的代码将在全部槽函数运行完成后被运行。
2.Qt::QueuedConnection(排队方式)(此时信号被塞到信号队列里了,信号与槽函数关系类似于消息通信。异步运行) 当信号发出后。排队到信号队列中,需等到接收对象所属线程的事件循环取得控制权时才取得该信号。调用对应的槽函数。emit语句后的代码将在发出信号后马上被运行。无需等待槽函数运行完成。
3.Qt::AutoConnection(自己主动方式) Qt的默认连接方式,假设信号的发出和接收这个信号的对象同属一个线程,那个工作方式与直连方式同样。否则工作方式与排队方式同样。
4.Qt::BlockingQueuedConnection(信号和槽必须在不同的线程中。否则就产生死锁) 这个是全然同步队列仅仅有槽线程运行完毕才会返回。否则发送线程也会一直等待,相当于是不同的线程能够同步起来运行。
5.Qt::UniqueConnection 与默认工作方式同样。仅仅是不能反复连接同样的信号和槽。由于假设反复连接就会导致一个信号发出。相应槽函数就会运行多次。
6.Qt::AutoCompatConnection 是为了连接Qt4与Qt3的信号槽机制兼容方式。工作方式与Qt::AutoConnection一样。 假设这个參数不设置的话。默认表示的是那种方式呢? 没加的话与直连方式同样:当信号发出后,对应的槽函数将马上被调用。emit语句后的代码将在全部槽函数运行完成后被运行。
在这个线程内是顺序运行、同步的。可是与其他线程之间肯定是异步的了。假设使用多线程,仍然须要手动同步。

在一般情况下,线程使用多线程异步方式还是同步方式好一些?

在一般情况下,使用多线程异步方式通常更好。异步方式可以提高程序的性能和响应速度,因为它允许程序在等待某些操作完成时继续执行其他操作,而不会阻塞整个程序。这在需要进行网络请求、IO操作或其他耗时操作时特别有用。
相比之下,同步方式可能会导致程序在等待某些操作完成时出现阻塞,影响整体性能。但是需要注意,多线程编程也可能引入一些复杂性和潜在的并发问题,如竞态条件和死锁等,因此在设计和实现多线程时需要谨慎考虑这些因素。

  • 14
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Qt Modbus是一个用于与Modbus设备通信的库。同步读异步写是指在读操作时采用同步方式,而在写操作时采用异步方式。 在同步读操作中,主线程会等待从Modbus设备读取到数据后才会继续执行后续操作。这意味着主线程会被阻塞,直到读取操作完成。同步读取可以保证在读取到数据后立即进行相应的处理,但如果读取操作时间较长,可能会导致用户界面无响应或造成其他延迟。 异步写操作是指主线程在发送写请求后会立即继续执行后续代码,而不必等待写操作完成。写操作会在后台线程中进行,这样可以避免阻塞主线程,并允许主线程继续执行其他任务。异步写操作提供了较好的响应性能和用户体验。 使用Qt Modbus进行同步读异步写时,可以通过在读取操作时使用QModbusTcpClient的read函数,并在写操作时使用QModbusTcpClient的sendWriteRequest函数来实现。 例如,可以使用类似以下的代码来实现同步读异步写: ```cpp // 创建Modbus Tcp Client对象 QModbusTcpClient *client = new QModbusTcpClient(this); // 连接到Modbus设备 client->setConnectionParameter(QModbusDevice::NetworkPortParameter, 502); client->setConnectionParameter(QModbusDevice::NetworkAddressParameter, "192.168.1.1"); client->connectDevice(); // 同步读操作 if (auto *reply = client->sendReadRequest(readRequest, serverAddress)) { if (!reply->isFinished()) QEventLoop loop; connect(reply, &QModbusReply::finished, &loop, &QEventLoop::quit); loop.exec(); if (reply->error() == QModbusDevice::NoError) { // 根据读取到的数据进行相应的处理 } else { // 处理读取错误 } reply->deleteLater(); } else { // 处理读取请求错误 reply->deleteLater(); } } // 异步写操作 if (auto *reply = client->sendWriteRequest(writeRequest, serverAddress)) { connect(reply, &QModbusReply::finished, this, &MyClass::writeFinished); reply->deleteLater(); } ``` 通过以上代码,我们可以实现在读操作时等待数据的同步读取,而在写操作时继续执行其他任务的异步写入。这样可以提高程序的响应性和用户体验。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值