一、QProcess介紹
QProcess類用于啟動外部程式并與它們通信。
QProcess允許将程序視為I/O裝置。可以像使用qtcsocket通路網絡連接配接一樣對程序進行寫入和讀取。然後可以通過調用write()寫入程序的标準輸入,并通過調用read()、readLine()和getChar()讀取标準輸出。因為QProcess繼承了QIODevice,是以它還可以用作QXmlReader的輸入源,或者用于生成要使用QNetworkAccessManager上載的資料。
當程序退出時,QProcess重新進入NotRunning狀态(初始狀态),并發出finished()。
finished()信号提供程序的退出代碼和退出狀态作為參數,還可以調用exitCode()來擷取最後一個完成的程序的退出代碼,并調用exitStatus()來擷取其退出狀态。如果在任何時間點發生錯誤,QProcess将發出errorOccurred()信号。還可以調用error()來查找上次發生的錯誤類型,調用state()來查找目前程序狀态。
程序有兩個預定義的輸出通道:标準輸出通道(stdout)提供正常控制台輸出,标準錯誤通道(stderr)通常提供程序列印的錯誤。這些通道代表兩個獨立的資料流。可以通過調用setReadChannel()在它們之間切換。目前讀取通道上有可用資料時,QProcess發出readyRead()。當新的标準輸出資料可用時,它還發出readyReadStandardOutput(),當新的标準錯誤資料可用時,發出readyReadStandardError()。不必調用read()、readLine()或getChar(),可以通過調用readAllStandardOutput()或readAllStandardError()顯式讀取兩個通道中的任何一個通道的所有資料。
QProcess提供了一組函數,允許在沒有事件循環的情況下使用它,方法是挂起調用線程,直到發出某些信号:
waitForStarted()會一直阻塞,直到程序啟動。
waitForReadyRead()阻塞,直到新資料可用于目前讀取通道上的讀取。
waitForBytesWrite()阻塞,直到一個有效負載的資料被寫入程序。
waitForFinished()阻塞,直到程序完成。
從主線程(調用QApplication::exec()的線程)調用這些函數可能會導緻使用者界面當機。
下面通過幾個例子介紹QProcess的使用場景和方法。
1. 第一個例子調用ipconfig指令擷取本地IP資訊,示範如何阻塞執行指令并得到指令的輸出,并解決輸出的中文亂碼問題。
2. 第二個例子調用ffmpge擷取視訊檔案的資訊,示範如何阻塞執行指令并得到指令的輸出。
3. 第三個例子調用ping指令ping百度,擷取網絡連接配接情況,示範如何實時擷取指令的輸出。
4. 第四個例子調用ffmpge指令完成視訊轉碼,示範如何實時擷取指令的輸出,并寫資料給程序,完成互動--->就是如何中途正常的退出ffmpge指令的執行。
工程下載下傳位址:
https://download.csdn.net/download/xiaolong1126626497/20632376 二、使用示例: windows下調用ipconfig擷取系統IP#include <QProcess>
#include <QTextCodec>
QProcess process;
process.start("ipconfig");
process.waitForFinished();
process.waitForReadyRead();
QByteArray qba = process.readAll();
//解決中文亂碼問題
QTextCodec* pTextCodec = QTextCodec::codecForName("System");
assert(pTextCodec != nullptr);
QString str = pTextCodec->toUnicode(qba);
qDebug("%s\n",str.toStdString().c_str());
輸出結果:
Windows IP 配置
無線區域網路擴充卡 WLAN:
連接配接特定的 DNS 字尾 . . . . . . . :
本地連結 IPv6 位址. . . . . . . . : fe80::f887:2337:ca8f:e8d5%10
IPv4 位址 . . . . . . . . . . . . : 10.0.0.4
子網路遮罩 . . . . . . . . . . . . : 255.255.255.0
預設網關. . . . . . . . . . . . . : 10.0.0.1
以太網擴充卡 以太網:
媒體狀态 . . . . . . . . . . . . : 媒體已斷開連接配接
連接配接特定的 DNS 字尾 . . . . . . . :
無線區域網路擴充卡 本地連接配接* 1:
媒體狀态 . . . . . . . . . . . . : 媒體已斷開連接配接
連接配接特定的 DNS 字尾 . . . . . . . :
無線區域網路擴充卡 本地連接配接* 4:
媒體狀态 . . . . . . . . . . . . : 媒體已斷開連接配接
連接配接特定的 DNS 字尾 . . . . . . . :
以太網擴充卡 VMware Network Adapter VMnet1:
連接配接特定的 DNS 字尾 . . . . . . . :
本地連結 IPv6 位址. . . . . . . . : fe80::5c33:8a5b:a8a6:3026%19
IPv4 位址 . . . . . . . . . . . . : 192.168.112.1
子網路遮罩 . . . . . . . . . . . . : 255.255.255.0
預設網關. . . . . . . . . . . . . :
以太網擴充卡 VMware Network Adapter VMnet8:
連接配接特定的 DNS 字尾 . . . . . . . :
本地連結 IPv6 位址. . . . . . . . : fe80::754a:6573:6487:c8f0%18
IPv4 位址 . . . . . . . . . . . . : 192.168.24.1
子網路遮罩 . . . . . . . . . . . . : 255.255.255.0
預設網關. . . . . . . . . . . . . :
三、使用示例: 調用ffmpge檢視視訊檔案資訊
#include <QProcess>
#include <QTextCodec>
QProcess process;
process.start("D:\\linux-share-dir\\C++_v5\\ECRS_Object\\Release\\ffprobe.exe -v quiet -of json -i D:/123.mp4 -show_streams ");
process.waitForFinished();
process.waitForReadyRead();
QByteArray qba = process.readAll();
QTextCodec* pTextCodec = QTextCodec::codecForName("System");
assert(pTextCodec != nullptr);
QString str = pTextCodec->toUnicode(qba);
qDebug("%s\n",str.toStdString().c_str());
{
"streams": [
{
"index": 0,
"codec_name": "aac",
"codec_long_name": "AAC (Advanced Audio Coding)",
"profile": "LC",
"codec_type": "audio",
"codec_time_base": "1/88200",
"codec_tag_string": "mp4a",
"codec_tag": "0x6134706d",
"sample_fmt": "fltp",
"sample_rate": "88200",
"channels": 2,
"channel_layout": "stereo",
"bits_per_sample": 0,
"r_frame_rate": "0/0",
"avg_frame_rate": "0/0",
"time_base": "1/44100",
"start_pts": 0,
"start_time": "0.000000",
"duration_ts": 4142070,
"duration": "93.924490",
"bit_rate": "127916",
"max_bit_rate": "132760",
"nb_frames": "4045",
"disposition": {
"default": 1,
"dub": 0,
"original": 0,
"comment": 0,
"lyrics": 0,
"karaoke": 0,
"forced": 0,
"hearing_impaired": 0,
"visual_impaired": 0,
"clean_effects": 0,
"attached_pic": 0,
"timed_thumbnails": 0
},
"tags": {
"creation_time": "2015-04-30T02:43:22.000000Z",
"language": "und",
"handler_name": "GPAC ISO Audio Handler"
}
},
{
"index": 1,
"codec_name": "h264",
"codec_long_name": "H.264 / AVC / MPEG-4 AVC / MPEG-4 part 10",
"profile": "Main",
"codec_type": "video",
"codec_time_base": "2349/70450",
"codec_tag_string": "avc1",
"codec_tag": "0x31637661",
"width": 1280,
"height": 720,
"coded_width": 1280,
"coded_height": 720,
"has_b_frames": 0,
"sample_aspect_ratio": "1:1",
"display_aspect_ratio": "16:9",
"pix_fmt": "yuv420p",
"level": 51,
"chroma_location": "left",
"refs": 1,
"is_avc": "true",
"nal_length_size": "4",
"r_frame_rate": "25/1",
"avg_frame_rate": "35225/2349",
"time_base": "1/30000",
"start_pts": 0,
"start_time": "0.000000",
"duration_ts": 2818800,
"duration": "93.960000",
"bit_rate": "581978",
"bits_per_raw_sample": "8",
"nb_frames": "1409",
"disposition": {
"default": 1,
"dub": 0,
"original": 0,
"comment": 0,
"lyrics": 0,
"karaoke": 0,
"forced": 0,
"hearing_impaired": 0,
"visual_impaired": 0,
"clean_effects": 0,
"attached_pic": 0,
"timed_thumbnails": 0
},
"tags": {
"creation_time": "2015-04-30T02:43:23.000000Z",
"language": "und",
"handler_name": "GPAC ISO Video Handler"
}
}
]
}
四、使用示例: 調用ping指令擷取實時輸出
想要實時擷取process的标準輸出,需要關聯readyReadStandardOutput信号;并且process需要動态的new出來。
4.1 做了個簡單的ui界面
4.2 cpp檔案代碼
#ifndef WIDGET_H
#define WIDGET_H
#include <QProcess>
#include <QTextCodec>
#include <QWidget>
QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE
class Widget : public QWidget
{
Q_OBJECT
public:
Widget(QWidget *parent = nullptr);
~Widget();
QProcess *process;
private slots:
void slot_readyRead();
void on_pushButton_start_clicked();
void on_pushButton_stop_clicked();
void on_pushButton_exit_clicked();
private:
Ui::Widget *ui;
};
#endif // WIDGET_H
4.3 .h檔案代碼
#ifndef WIDGET_H
#define WIDGET_H
#include <QProcess>
#include <QTextCodec>
#include <QWidget>
QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE
class Widget : public QWidget
{
Q_OBJECT
public:
Widget(QWidget *parent = nullptr);
~Widget();
QProcess *process;
private slots:
void slot_readyRead();
void on_pushButton_start_clicked();
void on_pushButton_stop_clicked();
void on_pushButton_exit_clicked();
private:
Ui::Widget *ui;
};
#endif // WIDGET_H
4.4 運作效果
五、使用示例: 調用ffmpge指令完成視訊轉碼
下面的例子示範如何調用ffmpge指令完成視訊轉碼,并且實時擷取轉碼的進度輸出,解析之後可以制作進度條界面,還可以向程序寫指令進去(寫q可以中斷ffmpge的執行,正常儲存退出),與ffmpge程序互動。
5.1 UI界面
5.2 widget.h 代碼
#ifndef WIDGET_H
#define WIDGET_H
#include <QProcess>
#include <QTextCodec>
#include <QWidget>
QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE
class Widget : public QWidget
{
Q_OBJECT
public:
Widget(QWidget *parent = nullptr);
~Widget();
QProcess *process;
private slots:
void slot_readyRead();
void on_pushButton_start_clicked();
void on_pushButton_stop_clicked();
void on_pushButton_exit_clicked();
private:
Ui::Widget *ui;
};
#endif // WIDGET_H
5.3 widget.cpp代碼
#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
Widget::Widget(QWidget *parent)
: QWidget(parent)
, ui(new Ui::Widget)
{
ui->setupUi(this);
process=new QProcess(this);
QObject::connect(process,SIGNAL(readyReadStandardOutput()),this, SLOT(slot_readyRead()));
process->setProcessChannelMode(QProcess::MergedChannels);
}
Widget::~Widget()
{
delete ui;
}
/*
工程: untitled1
日期: 2021-07-28
作者: DS小龍哥
環境: win10 QT5.12.6 MinGW32
功能: 啟動轉碼
*/
void Widget::on_pushButton_start_clicked()
{
//process->start("C:/FFMPEG/ffmpeg_x86_4.2.2/bin/ffmpeg.exe -i \"D:/test1080.flv\" -y -qscale 0 -vcodec libx264 -acodec aac -ac 1 -ar 22050 -b:v 0 -s 1280x720 -r 25 \"D:/linux-share-dir/video_file/test/out.mp4\"");
process->start(ui->lineEdit_start->text());
}
/*
工程: untitled1
日期: 2021-07-28
作者: DS小龍哥
環境: win10 QT5.12.6 MinGW32
功能: 有資料可讀
*/
void Widget::slot_readyRead()
{
QByteArray qba = process->readAllStandardOutput();
QTextCodec* pTextCodec = QTextCodec::codecForName("System");
assert(pTextCodec != nullptr);
QString str = pTextCodec->toUnicode(qba);
ui->plainTextEdit->insertPlainText(str);
}
/*
工程: untitled1
日期: 2021-07-28
作者: DS小龍哥
環境: win10 QT5.12.6 MinGW32
功能: 寫資料
*/
void Widget::on_pushButton_stop_clicked()
{
process->write(ui->lineEdit_write->text().toLocal8Bit());
}
/*
工程: untitled1
日期: 2021-07-28
作者: DS小龍哥
環境: win10 QT5.12.6 MinGW32
功能: 停止指令
*/
void Widget::on_pushButton_exit_clicked()
{
process->close();
process->waitForFinished();
}
5.4 程式運作效果