我为我的应用程序设计了一个GUI,点击一下按钮后,我连接了我的一个自定义函数,其中我调用了一个bash命令来同时运行2个Python脚本。这两个脚本都是Qt应用程序。我有一个名为'Start'的按钮来运行bash命令。但是一旦我点击了按钮,两个应用程序都会启动,但我的GUI会冻结。此外,我有一个“停止”按钮来停止这些应用程序,但GUI冻结,我无法对GUI做任何事情。我究竟做错了什么?
我已经尝试了同时运行2个脚本的各种选项,但命令python3 script1.py & python3 script2.py &
对我来说很好。还有其他方法吗? os.system()
或subprocess.call()
对我不起作用。有没有有效的方法呢?
这是我的UI的主要代码。
class MainApp(QtWidgets.QMainWindow, mainui.Ui_MainWindow):
def __init__(self, parent=None):
super(MainApp, self).__init__(parent)
self.setupUi(self)
self.exitapp()
self.startApp()
def multipleopen(self):
self.output = subprocess.check_output(['bash', '-c',
"python3 guiGO.py & python3 liveSpectogram.py &"])
def startApp(self):
self.start.clicked.connect(self.multipleopen)
def exitapp(self):
self.exit.clicked.connect(QtCore.QCoreApplication.instance().quit)
我希望在单击“开始”按钮时启动两个应用程序,然后单击“停止”按钮停止它。但点击“开始”按钮后,GUI冻结,除了关闭正在运行的应用程序并关闭GUI窗口,我无法做任何事情
GUI变得反应迟钝
我要强制退出GUI
我必须使用停止按钮手动停止脚本
函数subprocess.check_output()
是阻塞的,所以它会阻止GUI的事件循环通过冻结你的应用程序来完成它的工作,而你应该使用QProcess
:
class MainApp(QtWidgets.QMainWindow, mainui.Ui_MainWindow):
def __init__(self, parent=None):
super(MainApp, self).__init__(parent)
self.setupUi(self)
self._process = QtCore.QProcess(self)
self._process.readyReadStandardOutput.connect(self.on_readyReadStandardOutput)
self._process.setProcessChannelMode(QtCore.QProcess.ForwardedChannels)
self._process.setProgram('bash')
self._process.setArguments(['-c', 'python3 guiGO.py & python3 liveSpectogram.py &'])
self.exitapp()
self.startApp()
# read prints
def on_readyReadStandardOutput(self):
self.output = self._process.readAllStandardOutput()
print(self.output)
def startApp(self):
self.start.clicked.connect(self._process.start)
def exitapp(self):
self.exit.clicked.connect(self.close)
更新:
如果你想杀死python脚本,你必须使用pkill命令通过bash执行它,因为这是启动它们的那个。在代码的哪一部分你应该调用pkill ?,可能的部分是closeEvent方法。
from PyQt5 import QtCore, QtWidgets
class MainApp(QtWidgets.QMainWindow): #, mainui.Ui_MainWindow):
def __init__(self, parent=None):
super(MainApp, self).__init__(parent)
self.setupUi(self)
self._scripts = ("guiGO.py", "liveSpectogram.py")
self._processes = []
for script in self._scripts:
process = QtCore.QProcess(self)
process.readyReadStandardOutput.connect(self.on_readyReadStandardOutput)
process.setProcessChannelMode(QtCore.QProcess.ForwardedChannels)
process.setProgram('bash')
process.setArguments(['-c', 'python3 {}'.format(script)])
self._processes.append(process)
self.exitapp()
self.startApp()
# read prints
def on_readyReadStandardOutput(self):
self.output = self.sender().readAllStandardOutput()
print(self.output)
def startApp(self):
for process in self._processes:
self.start.clicked.connect(process.start)
def exitapp(self):
self.exit.clicked.connect(self.close)
def closeEvent(self, event):
for script in self._scripts:
QtCore.QProcess.startDetached("bash", ["-c", "pkill -f {}".format(script)])
super(MainApp, self).closeEvent(event)
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainApp()
w.show()
sys.exit(app.exec_())