本文主要是介绍QProcess类实现将子进程的窗口嵌入的主进程中,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
在主进程中创建子进程,子进程也是通过qt新建的qwidget窗口的应用程序.
bool CncOpWindows::LoadVisionMeasureApp()
{QString cmd= QCoreApplication::applicationDirPath();int iCamAppType = 0;switch (iCamAppType){case 0:cmd += "/ImageGrab.exe";//子程序执行文件地址break;case 5:cmd += "/MeasureVM.exe";//子程序执行文件地址break;}QStringList argList;
//将子进程的窗口嵌入stackedWidget对象中,所以将父对象的id传给子进程.argList << QString::number(ui->stackedWidget_sub->winId());//把父窗口的id给子进程传递过去m_pProcessVM = new QProcess(this);//使用进程运行子进程窗口connect(m_pProcessVM, &QProcess::readyReadStandardError, this, &CncOpWindows::slotCreateWaitingVM);//等待子进程窗口把自身的winId传递过来,在子进程中需要做对应的动作
//响应子进程结束的消息,查询子进程是否正常结束.connect(m_pProcessVM, qOverload<int, QProcess::ExitStatus>(&QProcess::finished), this, qOverload<int , QProcess::ExitStatus>(&CncOpWindows::slotFinishedProcessVM));//connect(m_pProcessVM, &QProcess::readyRead, this, &CncOpWindows::slotReadProcessVM);//开启子进程程序m_pProcessVM->start(cmd, argList);
//等待2000ms,返回false说明启动子进程失败.return m_pProcessVM->waitForStarted(2000);
}
其他槽函数,及析构时的函数
void CncOpWindows::slotCreateWaitingVM()
{
//子进程返回的错误码,为子进程的窗口idquint64 winId = m_pProcessVM->readAllStandardError().toLongLong();
//根据子进程的窗口id, 构造一个QWindow对象指针.m_ChildVMWin = QWindow::fromWinId(winId);if (m_ChildVMWin){
//通过这个QWindow对象指针,通过QWidget::createWindowContainer的静态函数,封装到这个Qwidget容器中.m_WidgetVMProcess = QWidget::createWindowContainer(m_ChildVMWin);//获取一个子进程窗口的widget//将这个QWidget对象指针,添加到stackedWidget_sub中.ui->stackedWidget_sub->addWidget(m_WidgetVMProcess);//这里是可以使用布局器管理子进程窗口的,不管理的话就在坐标0,0处}}void CncOpWindows::slotFinishedProcessVM(int exitCode, QProcess::ExitStatus exitStatus)
{
//进程结束时的状态判断qInfo() << "ImageGrab.exe quit"<<"[exitCode="<< exitCode << "ExitStatus=" << exitStatus << "]";//if (exitStatus== QProcess::NormalExit)//{// //正常退出//}//else//{// //非正常退出//}}//这个函数一定要放到析构函数里面,不能放到closeevent事件中处理.会导致提前结束进程.
void CncOpWindows::CloseVisionMeasureApp()
{//关闭当前视觉测量进程if (m_pProcessVM){////关键点,必须将嵌入的QWindow设置为没有父对象,在调用退出指令就正常了if (m_ChildVMWin){m_ChildVMWin->setParent(nullptr);}//正常退出程序terminate 发送一个close消息到顶层窗口.m_pProcessVM->terminate();//等待结束不能省,bool bFinish=m_pProcessVM->waitForFinished(5000);if (!bFinish){//非正常退出qInfo() << "ImageGrab.exe crash quit";}}}
子进程中相应处理:
在main函数中添加如下
int main(int argc, char *argv[])
{QApplication a(argc, argv);ImageGrab w;if (argc >1){WId wid = WId(QString(argv[1]).toInt());//通过参数列表获取父进程窗口的WinIdQWindow* window = QWindow::fromWinId(wid);//获取父进程窗口w.setProperty("_q_embedded_native_parent_handle", QVariant(wid));//设置属性,这句是必须的w.winId();//必须调用一次,生成winIdw.windowHandle()->setParent(window);//设置父窗口//关键代码,触发错误之后,主进程能够响应fprintf(stderr, "%lld", (qint64)w.winId());//写入标准错误输出,stderr能立即输出,stdout则不行w.m_bSubProcess = true;}else{w.m_bSubProcess = false;}w.show();return a.exec();
}
这篇关于QProcess类实现将子进程的窗口嵌入的主进程中的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!