子线程中的 waitForReadyRead() 冻结 GUI 线程
Posted
技术标签:
【中文标题】子线程中的 waitForReadyRead() 冻结 GUI 线程【英文标题】:waitForReadyRead() in subthread freezes the GUI thread 【发布时间】:2015-12-03 12:22:17 【问题描述】:对不起我的英语不好:(
我使用了一个继承自 QObject 的名为 SerialPort 的类:
serial_port.cpp & serial_port.h:
class SerialPort: public QObject
Q_OBJECT
private:
bool checkStatusConnected();
private slots:
void readData();
bool SerialPort::checkStatusConnected()
port.waitForReadyRead(200);
QBytesArray recv = port.readAll();
QString tmp = (QString)recv;
if(tmp.contains("Connected", Qt::CaseInsensitive))
return true;
return false;
MainWindow.cpp
void MainWindow::on_btnConn_clicked()
QThread* thread = new QThread(this);
serialport->moveToThread(thread);
serialport->connect();
在 SerialPort::connect() 中,我打开了端口并发送了连接命令。到现在一切正常。由于串口速度慢,我需要在下一步操作之前检查返回给我的字符串,所以在 SerialPort::connect() 中我调用 checkStatusConnected()。但是,在这个函数中,waitForReadyRead() 会冻结 GUI 线程,只要我设置的时间就会让我感到困惑。
GUI线程为什么会被另一个QThread阻塞以及如何解决这个问题?
谢谢!
【问题讨论】:
你的串口真的需要在单独的线程中运行吗? @ramtheconqueror 如果连接,我使用信号/插槽来防止 GUI 被冻结,但是连接时我想等待大约 500 毫秒的响应,然后发送数据或抛出错误。观看时,我希望 gui 正常工作,而不是被冻结。我怎么能意识到这一点? 您确定线程亲和性更改正确吗?serialport->thread()
与 this->thread()
相比返回了什么?
【参考方案1】:
您不需要为此使用线程。 QSerialPort
是事件驱动的并且异步工作。使用计时器来查看您是否在规定的时间内收到了东西。
QSerialPort port(....);
connect(&port, SIGNAL(readyRead()), this, SLOT(readFromSerialPort()));
QTimer timer;
timer.setInterVal(1000); // time you want to wait
connect(&timer, SIGNAL(timeout()), this, SLOT(handleTimeout()));
void SomeClass::readFromSerialPort()
QByteArray data(port.readAll());
// do somework
找到示例。看看这个,我认为这足以满足您的需求。
http://doc.qt.io/qt-5/qtserialport-creaderasync-example.html
【讨论】:
问题解决了!谢谢!我认为我应该在进行任何编码之前阅读更多文档。以上是关于子线程中的 waitForReadyRead() 冻结 GUI 线程的主要内容,如果未能解决你的问题,请参考以下文章
可以使用 `waitForReadyRead()` 而不是为 `readyRead()` 信号创建一个插槽吗?
使用QProcess,是不是需要在waitForFinished之后调用waitForReadyRead?
QSerialPort readyRead 信号仅在应用程序被 waitForReadyRead() 阻塞时发出