首先,我将从 QProcess 对象开始的简单 Qt 应用程序:
#include <QtGui>
class LOLProcess : public QMainWindow {
Q_OBJECT
public:
LOLProcess(QWidget *parent = 0);
~LOLProcess();
protected:
void closeEvent(QCloseEvent *);
};
LOLProcess::LOLProcess(QWidget *parent) : QMainWindow(parent) {
}
LOLProcess::~LOLProcess() {
qDebug() << "~LOLProcess()";
}
void LOLProcess::closeEvent(QCloseEvent *) {
qDebug() << "closeEvent()";
}
int main(int argc, char *argv[]) {
QApplication a(argc, argv);
LOLProcess w;
w.show();
return a.exec();
}
现在,带有 QProcess 对象的 Qt 应用程序:
#include <QtGui>
class MainWindow : public QMainWindow {
Q_OBJECT
public:
MainWindow(QWidget *parent = 0);
QProcess process;
public slots:
void close_down();
};
MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) {
process.start("../../lolprocess-build-desktop-Qt_4_8_1_for_Desktop_-_MinGW__Qt_SDK__Release/release/lolprocess.exe");
QTimer::singleShot(3000, this, SLOT(close_down()));
}
void MainWindow::close_down() {
process.terminate();
while (!process.waitForFinished(500)) {
qDebug() << process.readAllStandardError();
qDebug() << process.readAllStandardOutput();
}
qDebug() << process.waitForFinished(5000);
qDebug() << (process.state() == QProcess::Running);
}
int main(int argc, char *argv[]) {
QApplication a(argc, argv);
QDir::setCurrent(QCoreApplication::applicationDirPath());
MainWindow w;
w.show();
return a.exec();
}
行为:terminate() 导致子进程的窗口消失(并从 XP 任务栏中),但根据 taskman,该进程仍在运行。由于while (!process.waitForFinished(500))
QProcess 对象从不发出finished(),所以循环永远存在。令人惊讶的是,我从子进程(它只打印QProcess::readAllStandard...()
调用的空字符串)中没有得到任何关于 stdin 或 stderr 的信息,这看起来很奇怪,因为子进程的顶部窗口关闭让我认为 closeEvent() 在子进程中被调用。
使用 QProcess::kill() 终止子进程是可行的,但我不想使用它,因为它不会让子进程有机会在退出之前进行清理。
有趣的是,同样的代码在 Qt 4.7.4 和 OS X 10.6.x 下可以正常工作 - terminate() 导致子进程干净地退出,所以我不确定在 XP 下我做错了什么。谢谢。