Qtの基礎 - プロセス

提供:MochiuWiki : SUSE, EC, PCB
2024年2月10日 (土) 01:58時点におけるWiki (トーク | 投稿記録)による版 (→‎通常のプロセス)
ナビゲーションに移動 検索に移動

概要

Qtにおいて、QProcessクラスを使用して、プロセスを制御する手順を記載する。


通常のプロセス

プロセスを非同期で実行する場合はQProcess::start()、同期して実行する場合はQProcess.execute()を使用する。
また、非同期で実行する場合、標準出力や標準エラー出力を取得してウィンドウに表示することもできる。

以下の例は、プロセスを非同期で実行して、プロセスの標準出力と標準エラー出力をウィンドウに表示している。

 // MainWindow.h
 
 #ifndef MAINWINDOW_H
 #define MAINWINDOW_H
 
 #include <QMainWindow>
 #include <QProcess>
 
 QT_BEGIN_NAMESPACE
 namespace Ui { class MainWindow; }
 QT_END_NAMESPACE
 
 class MainWindow : public QMainWindow
 {
    Q_OBJECT
 
 private:
    QProcess m_proc;
  
 public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();
 
 private slots:
    // プロセス用のスロット
    void ProcessFinished(int exitCode, QProcess::ExitStatus exitStatus);
    void UpdateOutput();
    void UpdateError();
 };
 #endif // MAINWINDOW_H


 // MainWindow.cpp
 
 #include <QApplication>
 #include <QProcess>
 #include "Mainwindow.h"
 #include "ui_Mainwindow.h"
 
 MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow)
 {
    ui->setupUi(this);
 
    // プロセス用のシグナルとスロットを接続する
    // Qt 4の記述方法
    //QObject::connect(&m_proc, SIGNAL(readyReadStandardOutput()), this, SLOT(UpdateOutput()));
    //QObject::connect(&m_proc, SIGNAL(readyReadStandardError()), this, SLOT(UpdateError()));
    //QObject::connect(&m_proc, SIGNAL(finished(int, QProcess::ExitStatus)), this, SLOT(ProcessFinished(int, QProcess::ExitStatus)));

    // Qt 5の規約
    QObject::connect(&m_proc, &QProcess::readyReadStandardOutput, this, &MainWindow::UpdateOutput);
    QObject::connect(&m_proc, &QProcess::readyReadStandardError, this, &MainWindow::UpdateError);
    QObject::connect(&m_proc, static_cast<void(QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished), this, &MainWindow::ProcessFinished);
    
    // Qt 5 (C++14以降)
    //QObject::connect(&m_proc, qOverload<int, QProcess::ExitStatus >(&QProcess::finished), this, &MainWindow::ProcessFinished);
 }
 
 MainWindow::~MainWindow()
 {
    delete ui;
 }
 
 // スロット : 標準出力を表示する
 void MainWindow::UpdateOutput()
 {
    // 標準出力を取得して文字列にする
    QByteArray output = m_proc.readAllStandardOutput();
    QString str = QString::fromLocal8Bit(output);
 }
 
 // スロット : 標準エラー出力を表示する
 void MainWindow::UpdateError()
 {
    // 標準エラー出力を取得して文字列にする
    QByteArray output = m_proc.readAllStandardError();
    QString str = QString::fromLocal8Bit(output);
 }
 
 // スロット : プロセス終了時
 void MainWindow::ProcessFinished(int exitCode, QProcess::ExitStatus exitStatus)
 {
    if(exitStatus == QProcess::CrashExit)
    {
       QMessageBox::warning(this, tr("Error"), tr("Crashed"));
    }
    else if(exitCode != 0)
    {
       QMessageBox::warning(this, tr("Error"), tr("Failed"));
    }
    else
    {  // 正常終了時の処理
    }
 }
 
 // プロセスの起動
 void MainWindow::ProcessStart()
 {
    QStringList listArgs;
    listArgs << "<引数1>" << "<引数2>";
 
    m_proc.start("<実行するプログラムのパス>", listArgs);
 }


また、QProcessクラスのwaitForFinishedメソッドは、プロセスが終了するまで待機するメソッドである。
プロセスが正常に終了した場合はtrue、そうでない場合(操作のタイムアウト、エラーの発生、既にQProcessのオブジェクトが終了している)はfalseとなる。

 QProcess proc;
 proc.start("/usr/bin/sh" args);
 
 bool bStatus = proc.waitForFinished(10 * 1000);  // 最大10[s]待機
 if (m_proc.exitStatus() != QProcess::NormalExit || proc.exitCode() != 0)
 {
    std::fprintf(stderr, "error : %s", proc.readAllStandardError().constData());
    return -1;
 }



切り離されたプロセス

Qt 5.10以降、QProcessクラスを使用して、プロセスを切り離して実行できる。

QProcessクラスを使用してプロセスを切り離さずに実行する(startメソッドやexecuteメソッド等)場合、QProcessクラスのデストラクタがプロセスを終了する。
これに対して、切り離されたプロセスは、呼び出したプロセスが終了しても影響を受けずに実行し続ける。

Linuxでは、切り離されたプロセスは独自のセッションで実行されて、デーモンのように動作する。

切り離されたプロセスを実行するには、QProcess::startDetachedメソッドを使用する。
なお、QProcessクラスのstartDetachedメソッドは、静的メソッドであることに注意する。

以下の例では、aplayコマンドを使用して、Sample.wavを再生している。

 QProcess::startDetached("aplay Sample.wav");


通常のプロセスと同様に、複数の引数と作業ディレクトリを渡すことができる。
また、開始されたプロセスのPIDを取得することができる。

 // プロセスの開始
 qint64 pid;
 QProcess::startDetached("mpg123", {"Sample.mp3", <引数2>, <引数3>, ...}, musicDirPath, &pid);
 // ...略


QProcess::startDetachedメソッドで取得したPIDを使用して、プロセスをkillすることもできる。

 // プロセスの終了
 QProcess::startDetached("kill", {QString::number(pid)});


切り離されたプロセスを実行するにあたり、プロセスの出力を制御することもできる。
(使用するコマンドの中には、--quietオプションが存在するものもある)

  1. プロセス環境を設定する。
  2. stdinstdoutstderrをファイルにリダイレクトする。
  3. ネイティブな引数を設定、および、切り離されたプロセス用のCreateProcess関数(Windows API)の引数の設定


ただし、staticstartDetachedメソッドではなく、非staticQProcessクラスのstartDetachedメソッドを使用する必要がある。
これは、QProcessオブジェクトのプロパティを読み込み、それに応じて切り離されたプロセスを開始する。

以下の例では、mpg123コマンドを使用して、Sample.mp3を再生している。
/dev/nullへのリダイレクトは、ユーザがオーディオプレーヤの視覚的な出力を非表示にしている。

 QProcess process;
 
 process.setProgram("mpg123");
 process.setArguments({"Sample.mp3"});
 process.setWorkingDirectory(musicDirPath);  // 例. ファイルが存在するディレクトリのパス(/home/hpge/music等)
 
 process.setStandardOutputFile(QProcess::nullDevice());  // /dev/nullへのリダイレクト
 process.setStandardErrorFile(QProcess::nullDevice());   // /dev/nullへのリダイレクト
 
 qint64 pid;
 process.startDetached(&pid);
 
 // ...略


以下に、startDetachedメソッドでサポートされているサブセットを示す。
サブセットを使用する場合、QProcessクラスの他の全てのプロパティは無視されることに注意すること。

  • setArgumentsメソッド
  • setCreateProcessArgumentsModifierメソッド
  • setNativeArgumentsメソッド
  • setProcessEnvironmentメソッド
  • setProgramメソッド
  • setStandardErrorFileメソッド
  • setStandardInputFileメソッド
  • setStandardOutputFileメソッド
  • setWorkingDirectoryメソッド