使用matplotlib在Tkinter中绘制数据 - 在列表

时间:2015-10-19 19:08:20

标签: python class matplotlib tkinter interactive

我正在制作一个使用 Tkinter matplotlib 的程序。我有2个列表列表(一个用于x轴,一个用于y轴),我希望有一个可以在列表中的列表之间切换的按钮。我从问题Interactive plot based on Tkinter and matplotlib中获取了大部分代码,但我无法按照自己喜欢的方式使用该按钮。我很擅长使用课程并且理解它们有点困难。

tft是x数据 tf1是y数据

数据示例:

x-data = [[1,2,3,4,5],[10,11,13,15,12,19],[20,25,27]]
y-data = [[5.4,6,10,11,6],[4,6,8,34,20,12],[45,25,50]]

下面的代码将绘制列表中的一个列表,但是当我单击按钮时,不会在该列表中的列表之间切换。我已经评论了其他我试过的方法。当我使用它时,它总是说App没有属性“ line ”或“ canvas ”。就像我说的那样,我对课程很新,而且我想更好地理解它们。

我已经更新了我的代码,现在它可以识别event_num,并且只要按下按钮,它就会输出正确的值。但是,图表不是用新数据更新的(即它继续仅显示第一个数据集而不是列表之间的切换)。我认为问题出在函数increasedecrease中。我尝试使用self.line, = ax.plot(tft[self.event_num],tf1[self.event_num],'.')self.canvas.draw(),但它不起作用。我正在寻找要编辑的部分,以便图表发生变化。

class App: 


    def __init__(self, master):
        self.event_num = 1
        # Create a container
        frame = Frame(master)
        # Create 2 buttons
        self.button_left = Button(frame,text="< Previous Event",
                                        command=self.decrease)
        self.button_left.grid(row=0,column=0)
        self.button_right = Button(frame,text="Next Event >",
                                        command=self.increase)
        self.button_right.grid(row=0,column=1)

        fig = Figure()
        ax = fig.add_subplot(111)
        fig.autofmt_xdate()
        import matplotlib.dates as mdates
        ax.fmt_xdata = mdates.DateFormatter('%Y-%m-%d')
        self.line, = ax.plot(tft[self.event_num],tf1[self.event_num],'.')


        self.canvas = FigureCanvasTkAgg(fig,master=master)
        self.canvas.show()
        self.canvas.get_tk_widget().grid(row=1,column=0)
        frame.grid(row=0,column=0)

    def decrease(self):
        self.event_num -= 1
        print self.event_num
        self.line, = ax.plot(tft[self.event_num],tf1[self.event_num],'.')
        self.canvas.draw()
        #self.canvas.draw(tft[self.event_num],tf1[self.event_num],'.')
        #self.line.set_xdata(tft[event_num])
        #self.line.set_ydata(tf1[event_num])


    def increase(self):
        self.event_num += 1
        print self.event_num
        self.line, = ax.plot(tft[self.event_num],tf1[self.event_num],'.')
        self.canvas.draw()
        #self.canvas.draw(tft[self.event_num],tf1[self.event_num],'.')
        #self.set_xdata(tft[event_num])
        #self.set_ydata(tf1[event_num])


root = Tk()
app = App(root)
root.mainloop()

1 个答案:

答案 0 :(得分:1)

AttributeError: App instance has no attribute 'canvas'表示您的代码在创建/分配之前引用canvas属性。

这一行:

self.button_left = Button(frame,text="< Previous Event",
                                    command=self.decrease(event_num))

正在调用decrease方法,因为您使用了括号并提供了参数,而不是仅仅绑定处理程序。在reduce方法中,您正在访问self.canvas以调用draw方法。

在创建canvas属性之前会发生这种情况,该属性发生在这一行:

self.canvas = FigureCanvasTkAgg(fig,master=master)

使event_num成为App对象的属性;那么在绑定它时你不必将参数传递给处理程序。您可以在self.event_num = 1内分配__init__来完成此操作。