【问题标题】:PyQt5 doesn't launch QProcess in a window-chainPyQt5 不会在窗口链中启动 QProcess
【发布时间】:2022-10-26 02:08:15
【问题描述】:

我有一个程序需要多个脚本协同工作。

我设法用一个简化的例子重现了这种情况。

我有一个从另一个脚本(这里称为 LaunchTestv01)启动的主窗口(这里称为 StartTestUIv01),以便可以在我的实际脚本中添加变量和选项。这个主窗口有按钮,允许用户打开另一个窗口(这里是 BasicGraphUIv01),它本身使用另一个脚本(这里是 LaunchBasicGraphv01)获得的数据。

所以脚本链变成了这样: LaunchTestv01 -> StartTestUIv01 -> LaunchBasicGraphv01 -> BasicGraphUIv01

这是每个的代码: 启动测试v01

## This will launch StartTestUIv01 with the button
from PyQt5 import QtWidgets
from PyQt5.QtCore import QProcess
from StartTestUIv01 import Ui_MainWindow

class Window(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self, parent=None):
        super().__init__(parent=parent)
        self.setupUi(self)
        self.startGraph = QtWidgets.QAction('Launch Graph', self)
        
        self.w = None #Set the default value
        
        self.pushButton.clicked.connect(self.LaunchGraph)
    
    def LaunchGraph(self):
        if self.w is None:
            print("Working")
            # self.message("Executing Graph script")
            self.w = QProcess()
            self.w.finished.connect(self.closeGraph)
            self.w.start("python",['LaunchBasicGraphv01.py'])
    def closeGraph(self):
        print("Closing")
        # self.message("Graph script ended")
        self.w = None

if __name__ == '__main__':
    import sys
    app = QtWidgets.QApplication(sys.argv)
    window = Window()
    window.show()
    sys.exit(app.exec())

开始测试UIv01

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(313, 196)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setGeometry(QtCore.QRect(30, 20, 261, 121))
        self.pushButton.setObjectName("pushButton")
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 313, 22))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
        self.pushButton.setText(_translate("MainWindow", "START"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

LaunchBasicGraphv01

## This calculates values to insert in the Graph and displays that Window
from PyQt5 import QtWidgets
import BasicGraphUIv01 as ui
from pyqtgraph.Qt import QtCore
import numpy as np

x = np.arange(0,9)
y = np.random.randint(25, size=(9))

MainWindow = ui.QtWidgets.QMainWindow()
win = ui.Ui_MainWindow()
win.setupUi(MainWindow)
MainWindow.show()

# p0 = win.centralwidget.plot(x,y, pen='b')
p0 = win.graphWidget.plot(x,y, pen='y')

class Window(ui.QtWidgets.QMainWindow, ui.Ui_MainWindow):
    def __init__(self, parent=None):
        super().__init__(parent=parent)
        self.setupUi(self)

def update():
    y = np.random.randint(25, size=(9))
    p0.setData(x,y)

timer = QtCore.QTimer()
timer.timeout.connect(update)
timer.start(0)

基本图形UIv01

from PyQt5 import QtCore, QtGui, QtWidgets
from pyqtgraph import PlotWidget

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(473, 283)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.graphWidget = PlotWidget(self.centralwidget)
        self.graphWidget.setGeometry(QtCore.QRect(10, 10, 451, 211))
        self.graphWidget.setObjectName("graphWidget")
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 473, 22))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

现在明显的问题是链条没有启动! 似乎如果我正在启动一个本身正在启动另一个窗口的脚本,它就不会发生!

这样做的正确方法是什么?

编辑:所以我被禁止了,因为这个解决了问题不够好之类的。 澄清;在需要 python 脚本来收集数据的情况下,我正在寻找从另一个启动 PyQt 脚本的正确方法。

希望这可以帮助。

【问题讨论】:

  • 1. 使用 QProcess 时,捕捉错误总是很重要,否则你永远不会知道它是否(以及什么)没有按预期工作(例如,连接到errorOccurred 信号); 2. ui.QtWidgets.QMainWindow() in LaunchBasicGraphv01 完全没有意义: 1. ui 是一个完全不同进程的全局变量; 2.如果你想创建一个新的QMainWindow,那么就使用QtWidgets.QMainWindowui前缀完全没有意义; 3.那里没有创建QApplication,所以无论如何它都会崩溃。
  • @musicamante 我按照建议删除了 ui 变量,但即使没有更改,直接启动该脚本也可以正常工作!在那里添加一个 QApplication 的创建,就像在其他脚本上一样,会打开两个窗口(一个是空的)。如果我可以自己启动脚本就好了,为什么不从 QProcess 启动?我也尝试过“执行”而不是“开始”,结果相同。

标签: python pyqt5 subprocess qt-designer qprocess


【解决方案1】:

解决了!

感谢@musicamante 提供线索。

我只需要像这样更改两个启动脚本:

启动测试v01

## This will launch StartTestUIv01 with the button
from PyQt5 import QtWidgets
from PyQt5.QtCore import QProcess
from StartTestUIv01 import Ui_MainWindow

class Window(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self, parent=None):
        super().__init__(parent=parent)
        self.setupUi(self)
        self.startGraph = QtWidgets.QAction('Launch Graph', self)
        
        self.w = None #Set the default value
        
        self.pushButton.clicked.connect(self.LaunchGraph)
    
    def LaunchGraph(self):
        if self.w is None:
            print("Working")
            # self.message("Executing Graph script")
            self.w = QProcess()
            self.w.finished.connect(self.closeGraph)
            self.w.execute('python',['LaunchBasicGraphv01.py'])
    def closeGraph(self):
        print("Closing")
        # self.message("Graph script ended")
        self.w = None

if __name__ == '__main__':
    import sys
    app = QtWidgets.QApplication(sys.argv)
    window = Window()
    window.show()
    sys.exit(app.exec())

LaunchBasicGraphv01

from PyQt5 import QtWidgets
from BasicGraphUIv01 import Ui_MainWindow
from pyqtgraph.Qt import QtCore
import numpy as np

import sys
app = QtWidgets.QApplication(sys.argv)

x = np.arange(0,9)
y = np.random.randint(25, size=(9))

MainWindow = QtWidgets.QMainWindow()
win = Ui_MainWindow()
win.setupUi(MainWindow)
MainWindow.show()

# p0 = win.centralwidget.plot(x,y, pen='b')
p0 = win.graphWidget.plot(x,y, pen='y')

class Window(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self, parent=None):
        super().__init__(parent=parent)
        self.setupUi(self)

def update():
    y = np.random.randint(25, size=(9))
    p0.setData(x,y)

timer = QtCore.QTimer()
timer.timeout.connect(update)
timer.start(0)

if __name__ == '__main__':
    sys.exit(app.exec())

谢谢您的帮助 !

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多