我是Python的新手,我想请求帮助解决我的代码中的问题。我使用QtDesigner创建我的UI,然后使用pyuic4转换为test.py文件。我还有mplwidget.py和modules.py文件,如下所示:
test.py
from PyQt4 import QtCore, QtGui
from numpy import *
import threading, thread
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
try:
_fromUtf8 = QtCore.QString.fromUtf8
except AttributeError:
def _fromUtf8(s):
return s
try:
_encoding = QtGui.QApplication.UnicodeUTF8
def _translate(context, text, disambig):
return QtGui.QApplication.translate(context, text, disambig, _encoding)
except AttributeError:
def _translate(context, text, disambig):
return QtGui.QApplication.translate(context, text, disambig)
class Ui_MainWindow(object):
def setupUi(self, MainWindow):
MainWindow.setObjectName(_fromUtf8("MainWindow"))
MainWindow.resize(617, 588)
self.centralwidget = QtGui.QWidget(MainWindow)
self.centralwidget.setObjectName(_fromUtf8("centralwidget"))
self.verticalLayout = QtGui.QVBoxLayout(self.centralwidget)
self.verticalLayout.setObjectName(_fromUtf8("verticalLayout"))
self.tabWidget = QtGui.QTabWidget(self.centralwidget)
self.tabWidget.setObjectName(_fromUtf8("tabWidget"))
self.tab = QtGui.QWidget()
self.tab.setObjectName(_fromUtf8("tab"))
self.verticalLayout_2 = QtGui.QVBoxLayout(self.tab)
self.verticalLayout_2.setObjectName(_fromUtf8("verticalLayout_2"))
self.pushButton = QtGui.QPushButton(self.tab)
self.pushButton.setObjectName(_fromUtf8("pushButton"))
self.verticalLayout_2.addWidget(self.pushButton)
self.pushButton_2 = QtGui.QPushButton(self.tab)
self.pushButton_2.setObjectName(_fromUtf8("pushButton_2"))
self.verticalLayout_2.addWidget(self.pushButton_2)
self.tabWidget.addTab(self.tab, _fromUtf8(""))
self.tab_2 = QtGui.QWidget()
self.tab_2.setObjectName(_fromUtf8("tab_2"))
self.verticalLayout_3 = QtGui.QVBoxLayout(self.tab_2)
self.verticalLayout_3.setObjectName(_fromUtf8("verticalLayout_3"))
self.widget = MplWidget(self.tab_2)
sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Expanding)
sizePolicy.setHorizontalStretch(0)
sizePolicy.setVerticalStretch(0)
sizePolicy.setHeightForWidth(self.widget.sizePolicy().hasHeightForWidth())
self.widget.setSizePolicy(sizePolicy)
self.widget.setObjectName(_fromUtf8("widget"))
self.verticalLayout_3.addWidget(self.widget)
self.tabWidget.addTab(self.tab_2, _fromUtf8(""))
self.verticalLayout.addWidget(self.tabWidget)
MainWindow.setCentralWidget(self.centralwidget)
self.statusbar = QtGui.QStatusBar(MainWindow)
self.statusbar.setObjectName(_fromUtf8("statusbar"))
MainWindow.setStatusBar(self.statusbar)
self.retranslateUi(MainWindow)
self.tabWidget.setCurrentIndex(0)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
def retranslateUi(self, MainWindow):
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow", None))
self.pushButton.setText(_translate("MainWindow", "PushButton I", None))
self.pushButton_2.setText(_translate("MainWindow", "PushButton II", None))
self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab), _translate("MainWindow", "Tab 1", None))
self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab_2), _translate("MainWindow", "Tab 2", None))
QtCore.QObject.connect(self.pushButton, QtCore.SIGNAL(_fromUtf8("pressed()")), self.ploting1)
QtCore.QObject.connect(self.pushButton_2, QtCore.SIGNAL(_fromUtf8("pressed()")), self.calling)
def calling(self):
from modules import MyForm
arg="yes",
tmain=threading.Thread(target=MyForm().ploting2,args=(arg))
tmain.start()
def ploting1(self,plot="yes"):
if plot != "no":
t = logspace(1E-6,1,132,base=1E-6)
y1 = exp(-t/1E-3)
self.widget.canvas.fig.clear()
self.widget.canvas.ax = self.widget.canvas.fig.add_axes([0.15, 0.15, 0.7, 0.7])
self.widget.canvas.ax.set_xlabel('t')
self.widget.canvas.ax.set_ylabel('corr')
label1='y1'
self.widget.canvas.ax.semilogx(t,y1,'ro-',label=label1)
self.widget.canvas.ax.legend(loc='lower left')
self.widget.canvas.ax.tick_params(axis='y', colors='red')
self.widget.canvas.draw()
self.tabWidget.setCurrentIndex(1)
from mplwidget import MplWidget
if __name__ == "__main__":
import sys
app = QtGui.QApplication(sys.argv)
MainWindow = QtGui.QMainWindow()
ui = Ui_MainWindow()
ui.setupUi(MainWindow)
MainWindow.show()
sys.exit(app.exec_())
modules.py
from numpy import *
from test import Ui_MainWindow
from mplwidget import MplWidget
class MyForm(QtGui.QMainWindow, Ui_MainWindow):
def __init__(self, parent=None):
super(MyForm, self).__init__(parent)
self.setupUi(self)
def ploting2(self,plot="yes"):
if plot != "no":
t = logspace(1E-6,1,132,base=1E-6)
y1 = exp(-t/1E-3)
self.widget.canvas.fig.clear()
self.widget.canvas.ax = self.widget.canvas.fig.add_axes([0.15, 0.15, 0.7, 0.7])
self.widget.canvas.ax.set_xlabel('t')
self.widget.canvas.ax.set_ylabel('corr')
label1='y1'
self.widget.canvas.ax.semilogx(t,y1,'ro-',label=label1)
self.widget.canvas.ax.legend(loc='lower left')
self.widget.canvas.ax.tick_params(axis='y', colors='red')
self.widget.canvas.draw()
self.tabWidget.setCurrentIndex(1)
mplwidget.py
from PyQt4 import QtGui
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
class MplCanvas(FigureCanvas):
"""Class to represent the FigureCanvas widget"""
def __init__(self, parent=None, name=None, width=5, height=4, dpi=100, bgcolor=None):
self.parent = parent
if self.parent:
bgc = parent.backgroundBrush().color()
bgcolor = float(bgc.red())/255.0, float(bgc.green())/255.0, float(bgc.blue())/255.0
self.fig = Figure(figsize=(width, height), dpi=dpi, facecolor=bgcolor, edgecolor=bgcolor)
self.ax = self.fig.add_subplot(111)
self.ax.hold(False)
FigureCanvas.__init__(self, self.fig)
FigureCanvas.setSizePolicy(self,
QtGui.QSizePolicy.Expanding,
QtGui.QSizePolicy.Expanding)
FigureCanvas.updateGeometry(self)
class MplWidget(QtGui.QWidget):
"""Widget defined in Qt Designer"""
def __init__(self, parent = None):
QtGui.QWidget.__init__(self, parent)
self.canvas = MplCanvas()
self.vbl = QtGui.QVBoxLayout()
self.vbl.addWidget(self.canvas)
self.setLayout(self.vbl)
好吧,正如您在主窗口中看到的,我有两个按钮。第一个连接到test.py中定义的函数(ploting1),并在tab2中绘制图形。问题出现在按钮2中,我使用Thread调用modules.py中定义的函数(ploting2)。我正在使用Thread,因为我需要在绘制曲线的同时执行一些计算。两个函数都是相同的,但是从modules.py调用的函数没有显示图。对我做错了什么的暗示?
提前感谢您的帮助。
PS:我没有收到错误消息,一切正常,除了没有显示情节的按钮2。
答案 0 :(得分:1)
这里有几个问题。前两个围绕这条线:
tmain=threading.Thread(target=MyForm().ploting2,args=(arg))
tmain
是一个局部变量。 calling
方法完成后,
该线程将被垃圾收集。所以它不会运行MyForm()
调用的参数中实例化theading.Thread
,但不保存引用。到创建的对象。因此,对象中包含的GUI将被垃圾收集,并且将不复存在。所以你可以通过将变量存储在父对象中来解决这些问题(例如self.tmain = ...
等),但是从下一个问题中可以看出这是毫无意义的
我强烈建议您重新考虑使用线程。如果必须使用线程,请不要从线程中调用任何Qt方法(这包括matplotlib绘制调用)。
更新以回复评论
由于我进行了很长时间的计算,我认为最好的方法 正在使用线程。但我还想绘制参数 在计算过程中获得的实时你对这个有什么想法吗? 最好的方法吗?
您仍然可以在一个线程中进行长时间运行的计算,以便GUI的其余部分保持响应。然后,您需要将计算结果传达给主线程,您将在其中调用实际的绘图函数。您可以通过多种方式将数据传递到主线程。通过发出Qt信号(需要使用QThreads),明确地将Qt事件发布到主线程(使用QCoreApplication.postEvent()
或将数据放入Python队列。
一些相关的链接(不一定给出绘图示例,但应该给出一般的想法):