Qtの基礎 - プロセス
概要
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
オプションが存在するものもある)
- プロセス環境を設定する。
stdin
、stdout
、stderr
をファイルにリダイレクトする。- ネイティブな引数を設定、および、切り離されたプロセス用の
CreateProcess
関数(Windows API)の引数の設定
ただし、static
なstartDetached
メソッドではなく、非static
なQProcess
クラスの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
メソッド