如何关闭 Qt 子进程并让子进程执行清理代码?

Posted

技术标签:

【中文标题】如何关闭 Qt 子进程并让子进程执行清理代码?【英文标题】:How do you close a Qt child process and get the child process to execute cleanup code? 【发布时间】:2011-09-21 15:10:06 【问题描述】:

我正在 Linux/Qt 中启动一个进程,然后使用 QProcess 启动一些子进程。然后最终我想优雅地关闭子进程(也就是执行一些清理代码)。

子进程正在使用QSharedMemory,现在当我调用 QProcess::close() 时子进程正在关闭而不调用 QSharedMemory::detach(),结果是所有进程都关闭了......但是有剩余的未清理的共享内存。

我有子进程的代码,代码中有函数cleanup()。父进程如何关闭QProcess,让子进程执行cleanup()

【问题讨论】:

有很多不同的方法可以做到这一点。这里的目标是:我想看看 Qt 框架中使用的“常见”或“最佳实践”方法是什么。 仅供参考,所有进程都连接到同一个公共 dbus。 我不是 Qt 大师。所以我会保持这个开放状态,直到我得到有很多 Qt 经验的人的回复。 【参考方案1】:

我让孩子使用 unix 信号处理程序执行 Qt 清理代码。

这是一个高级解释:

父进程使用QProcess打开子进程 正在处理 父进程使用 QProcess::terminate() 关闭子进程,这会在子进程上引发 SIGTERM 信号 (不要使用QProcess::close() because it doesn't raise the SIGTERM signal) 孩子为 SIGTERM 实现了一个 unix 信号处理程序 来自 unix 信号处理程序的 qApp->exit(0);发生 qApp 发出 Qt 信号“aboutToQuit()” 将子进程 cleanup() 槽连接到 qApp aboutToQuit() 信号

处理 unix SIGTERM 信号的子进程代码:

static void unixSignalHandler(int signum) 
    qDebug("DBG: main.cpp::unixSignalHandler(). signal = %s\n", strsignal(signum));

    /*
     * Make sure your Qt application gracefully quits.
     * NOTE - purpose for calling qApp->exit(0):
     *      1. Forces the Qt framework's "main event loop `qApp->exec()`" to quit looping.
     *      2. Also emits the QCoreApplication::aboutToQuit() signal. This signal is used for cleanup code.
     */
    qApp->exit(0);


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

    QCoreApplication a(argc, argv);

    MAINOBJECT mainobject;

    /*
     * Setup UNIX signal handlers for some of the common signals.
     * NOTE common signals:
     *      SIGINT:     The user started the process on the command line and user ctrl-C.
     *      SIGTERM:    The user kills the process using the `kill` command.
     *                  OR
     *                  The process is started using QProcess and SIGTERM is
     *                  issued when QProcess::close() is used to close the process.
     */
    if (signal(SIGINT, unixSignalHandler) == SIG_ERR) 
        qFatal("ERR - %s(%d): An error occurred while setting a signal handler.\n", __FILE__,__LINE__);
    
    if (signal(SIGTERM, unixSignalHandler) == SIG_ERR) 
        qFatal("ERR - %s(%d): An error occurred while setting a signal handler.\n", __FILE__,__LINE__);
    
    // executes mainbobject.cleanupSlot() when the Qt framework emits aboutToQuit() signal.
    QObject::connect(qApp,          SIGNAL(aboutToQuit()),
                     &mainobject,   SLOT(cleanupSlot()));

    return a.exec();


结论:

我confirmed that this solution works。

我认为这是一个很好的解决方案,因为:

让父进程关闭子进程,让子进程执行清理 如果父进程错误关闭并让子进程继续运行,用户/系统管理员可以使用kill 命令杀死剩余的子进程,子进程在关闭前仍会自行清理

附注“为什么不直接在信号处理程序入口点做清理代码?”

简短的回答是因为你不能。这里解释了为什么你不能在 unix 信号处理函数中执行你的 Qt 清理代码。来自Qt documentation "Calling Qt Functions From Unix Signal Handlers":

您不能从 Unix 信号处理程序调用 Qt 函数。标准 POSIX 规则适用:您只能从 信号处理程序。有关功能的完整列表,请参阅 Signal Actions 您可以从 Unix 信号处理程序中调用。

【讨论】:

但我是 Qt 新手......所以我不确定这是否是常见的做事方式。或者也许有更优雅的解决方案。 在信号处理程序中调用 qApp->exit() 是不安全的 - 除非您审核代码以确认在您的平台上的实现中仅使用了 POSIX API 的异步安全子集。 【参考方案2】:

据我所知,您可以尝试将 processExited() 信号连接到一个插槽以自己处理与共享内存的分离,因为 QProcess 中没有任何内容会直接触发分离方法。

【讨论】:

子进程是必须调用执行cleanup()代码的那个。 你的建议是让父close() QProcess 并让父进程在QProcess 关闭时执行一些cleanup() 代码吗? 好吧,这是一个非此即彼的选择,我想说让父级在执行结束时关闭 QProcess 然后处理释放内存本身可能更安全,因为这样你就可以实现诸如保留之类的东西很重要,所以不可能发布需要的东西。 在这种特定情况下,父级无法进行清理。孩子必须这样做,因为孩子正在使用 [QSharedMemory](QSharedMemory):“Unix:QSharedMemory “拥有”共享内存段。当最后一个线程或进程将 QSharedMemory 实例附加到特定共享内存段时,从通过销毁其 QSharedMemory 实例,Unix 内核释放共享内存段。但如果最后一个线程或进程在没有运行 QSharedMemory 析构函数的情况下崩溃,则共享内存段在崩溃中幸存下来。" 啊哈哈,共享内存在哪里创建的?【参考方案3】:

我相信你需要在这里实现一个小协议。您需要一种方法来告诉子进程优雅地退出。如果您有这两者的来源,您可以尝试使用QtDBus 库来实现这样的信号。

【讨论】:

以上是关于如何关闭 Qt 子进程并让子进程执行清理代码?的主要内容,如果未能解决你的问题,请参考以下文章

Golang如何让子进程以另一个用户身份运行

子进程子回溯

如何让子进程等待其兄弟?

如何从 QT(C++)中的子进程联系父进程以执行类中的方法?

python多进程和多线程

shell多进程实例