0

我试图理解为什么不需要在我的 QProcess waitForFinished() 和 waitForStarted() 调用可以工作之前将 a.exec() 调用放入以下 ​​Qt 4.8 代码中。我知道 a.exec() 启动事件循环,在我看来,waitFor* 插槽需要在继续执行之前接收信号(即 'started()' 或 'finished()' )。如果事件循环尚未启动,怎么会发生这种情况?

waitForStarted() 的文档:

Blocks until the process has started and the started() signal has been emitted, or until msecs milliseconds have passed.

代码:

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

    QCoreApplication a(argc, argv);

    // Exec the i2c command to get the power button status
    QProcess i2cprocess;
    i2cprocess.start("/usr/bin/i2cget -f -y 1 0x4b 0x45");

    // Wait for it to start
    if(!i2cprocess.waitForStarted())
    {
        qDebug() << "Could not start QProcess to check power button status.";
        exit(-1);
    }

    // Wait for it to finish
    bool returnValue = i2cprocess.waitForFinished();
    if ( returnValue )
    {
        QByteArray status = i2cprocess.readAllStandardOutput().trimmed();
        bool ok;
        quint16 hexValue = status.toUInt(&ok, 16);
        qDebug() << "Power button status: " << status << hexValue << (hexValue & 0x01);

        // We want LSB
        exit(hexValue & 0x01);

    }
    else
    {
        qDebug() << "Error, process never completed to check power button status.";
        exit(-1);
    }

    return a.exec();

}
4

2 回答 2

3

直接信号槽连接只是间接函数调用,它们与事件循环没有任何关系。

但是,这些waitForXxxx方法所做的是旋转本地事件循环,直到信号触发。该信号是从一些代码中触发的,这些代码由操作系统通知进程状态已更改。该代码在功能上由事件循环“执行”。

请记住,在 Qt 中,您可以随心所欲地创建临时事件循环 - 这是一种不好的做法,您永远不应该编写使用waitFor方法的代码。它对您的代码提出了通常不存在的要求 - 从而引入了错误!

那么问题来了:在等待进程改变状态时,事件循环有什么用?在内部,流程通知需要等待本地事件或回调,并且这些都从事件循环中得到处理。即使没有使用任何事件,事件循环也会执行操作系统所需的可中断睡眠,以将回调传递给应用程序。

于 2015-05-29T15:42:08.390 回答
1

QProcess的 Qt 文档指出:-

QProcess 提供了一组函数,允许在没有事件循环的情况下使用它,方法是暂停调用线程直到发出某些信号:

waitForStarted() 阻塞,直到进程开始。

waitForReadyRead() 阻塞,直到有新数据可用于在当前读取通道上读取。

waitForBytesWritten() 阻塞,直到一个有效负载数据被写入进程。

waitForFinished() 阻塞,直到进程完成。

从主线程(调用 QApplication::exec() 的线程)调用这些函数可能会导致您的用户界面冻结。

于 2015-05-28T12:50:44.083 回答