线程中创建的图不会出现

时间:2014-11-21 11:08:58

标签: python matplotlib pyqt

我是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。

1 个答案:

答案 0 :(得分:1)

这里有几个问题。前两个围绕这条线:

tmain=threading.Thread(target=MyForm().ploting2,args=(arg))
  1. tmain是一个局部变量。 calling方法完成后, 该线程将被垃圾收集。所以它不会运行
  2. 您在MyForm()调用的参数中实例化theading.Thread,但不保存引用。到创建的对象。因此,对象中包含的GUI将被垃圾收集,并且将不复存在。
  3. 所以你可以通过将变量存储在父对象中来解决这些问题(例如self.tmain = ...等),但是从下一个问题中可以看出这是毫无意义的

    1. 您永远不应该从线程访问Qt小部件。这样做是expressly forbidden并且会在您的程序中随机导致可怕的可怕崩溃。
    2. 我强烈建议您重新考虑使用线程。如果必须使用线程,请不要从线程中调用任何Qt方法(这包括matplotlib绘制调用)。

      更新以回复评论

        

      由于我进行了很长时间的计算,我认为最好的方法   正在使用线程。但我还想绘制参数   在计算过程中获得的实时你对这个有什么想法吗?   最好的方法吗?

      您仍然可以在一个线程中进行长时间运行的计算,以便GUI的其余部分保持响应。然后,您需要将计算结果传达给主线程,您将在其中调用实际的绘图函数。您可以通过多种方式将数据传递到主线程。通过发出Qt信号(需要使用QThreads),明确地将Qt事件发布到主线程(使用QCoreApplication.postEvent()或将数据放入Python队列。

      一些相关的链接(不一定给出绘图示例,但应该给出一般的想法):