运行多个QProcess'组



我需要调用外部程序N多次。我想同时做这件事。因此,到目前为止,我的策略是启动NQ进程,并记录已启动和已完成的进程。(这样我就可以知道他们什么时候都结束了(。

然而,外部程序占用了相当多的RAM,所以我不希望在任何时候都有更多的4个并行进程。

对此,什么是好的策略?

我认为信号/插槽不足以实现这一点(我想不出一种方法不是高度复杂的(。。。也许我可以对排队做点什么?

如何确保一次只能运行4个进程?然后,我如何计算出所有N过程最终何时完成?

(首选pyside/pyqt的答案,但C++可以(

概念证明:

h文件

#ifndef CPROCESSRUNNER_H
#define CPROCESSRUNNER_H
#include <QObject>
#include <QQueue>
class QProcess;
class CProcessRunner : public QObject
{
    Q_OBJECT
    Q_PROPERTY(int processCount READ processCount WRITE setProcessCount)
public:
    explicit CProcessRunner(QObject *parent = 0);
    ~CProcessRunner();
    void addProcess(const QString& program);
    int processCount() const;
public slots:
    void setProcessCount(int arg);
private slots:
    void startFreeProcesses();
private:
    int getActiveProcessCount() const;
    QQueue<QProcess*> m_processes;
    int m_processCount;
};
#endif // CPROCESSRUNNER_H

cpp文件

#include "CProcessRunner.h"
#include <QProcess>
CProcessRunner::CProcessRunner(QObject *parent)
    : QObject(parent)
    , m_processCount(0)
{
}
CProcessRunner::~CProcessRunner()
{
}
void CProcessRunner::addProcess(const QString& program)
{
    QProcess* pProcess = new QProcess(this);
    pProcess->setObjectName(program);
    m_processes.enqueue(pProcess);
    startFreeProcesses();
}
int CProcessRunner::processCount() const
{
    return m_processCount;
}
void CProcessRunner::setProcessCount(int arg)
{
    m_processCount = arg;
}
void CProcessRunner::startFreeProcesses()
{
    while (!m_processes.isEmpty() && (getActiveProcessCount() < m_processCount)) {
        QProcess* pProcess = m_processes.dequeue();
        connect(pProcess, SIGNAL(finished(int)), this, SLOT(startFreeProcesses()));
        connect(pProcess, SIGNAL(finished(int)), pProcess, SLOT(deleteLater()));
        pProcess->start(pProcess->objectName());
        pProcess->waitForStarted(-1);
    }
}
int CProcessRunner::getActiveProcessCount() const
{
    int result = 0;
    foreach (QProcess* pProcess, findChildren<QProcess*>()) {
        if (pProcess->state() == QProcess::Running) {
            result++;
        }
    }
    return result;
}

为了参考,这里是我当前使用Pyside的实现。我希望能够捕获每个进程的输出并将其传递(例如在QTextEdit或类似的文件中显示(。为此,我为每个进程分配了一个数字,并将其stdout和stderr连接到插槽,然后插槽将发出消息和进程位置。我不相信我目前对该功能的实现,但这里是:

class ProcessRunner(QtCore.QObject):  
'''
Runs N processes in groups of M
'''    
singleProcessFinished = QtCore.Signal()
error = QtCore.Signal(tuple) #(process number, message)
message = QtCore.Signal(tuple) #(process number, message)
def __init__(self, maxProcesses=4, parent=None):
  super(ProcessRunner, self).__init__(parent)
  self.processQueue = Queue.Queue()  
  self.maxProcesses = maxProcesses
  self.activeProcessCount = 0
  self._processNumber = 0
def addProcess(self, program, args):
  '''
  Add a process to the process queue
  Args:
    program (str): String of program path and arguments, separated by one or more spaces    
  '''
  self._processNumber += 1
  if self._processNumber == self.maxProcesses:
    self._processNumber = 0

  proc = QtCore.QProcess(self)
  proc.readyReadStandardError.connect(lambda pos=self._processNumber, process=proc: self.emit_error(pos, proc))
  proc.readyReadStandardOutput.connect(lambda pos=self._processNumber, process=proc: self.emit_output(pos, proc))
  self.processQueue.put((proc, program, args))
  self.startFreeProcesses()
def startFreeProcesses(self):
  '''
  Starts all waiting processes up to a maximum of self.maxProcesses
  '''
  while (not self.processQueue.empty()) and (self.activeProcessCount < self.maxProcesses):
    proc, program, args = self.processQueue.get()
    proc.finished.connect(self.startFreeProcesses)
    proc.finished.connect(proc.deleteLater)
    proc.finished.connect(self.singleProcessFinished.emit)
    proc.finished.connect(self._decreaseActiveProcessCount)
    proc.start(program, args)
    proc.waitForStarted(-1)
    self._increaseActiveProcessCount()
def _decreaseActiveProcessCount(self):
  self.activeProcessCount -= 1
def _increaseActiveProcessCount(self):
  self.activeProcessCount += 1
def emit_error(self, pos, proc):    
  self.error.emit(pos, str(proc.readAllStandardError()))
def emit_error(self, pos, proc):    
  self.error.emit(pos, str(proc.readAllStandardOutput()))

相关内容

  • 没有找到相关文章

最新更新