PyQt5 matplotlib畫圖不刷新的解決方案
本人最近在做一個GUI項目時,需要用到matplotlib畫圖,寫完代碼運行時,發現所需要畫圖功能不能正常使用,在觸發畫圖事件後,畫佈未更新,需要放大界面才能更新(奇葩吧)。
經過資料查閱,發現需要畫圖函數後面添加
self.fig.canvas.draw_idle()
補充:解決PyQt5中使用柱狀圖和餅狀圖刷新問題
一、問題描述
在做一款番茄鐘應用,其中包含數據統計功能,如下圖:
我遇到的問題是:當完成一個番茄鐘後,需要對統計數據進行刷新,但是重新調用一下畫圖函數無效,下面貼出調用的這個畫圖函數:
class MyFigure(FigureCanvas, QWidget): def __init__(self,parent=None, minWidth=600, minHeight=380, dpi=120): #第一步:創建一個創建Figure self.fig = Figure(figsize=(8, 8), dpi=dpi) #第二步:在父類中**Figure窗口 super(MyFigure,self).__init__(self.fig) #此句必不可少,否則不能顯示圖形 #第三步:創建一個子圖,用於繪制圖形用,111表示子圖編號,如matlab的subplot(1,1,1) self.axes = self.fig.add_subplot(111) self.setMinimumSize(minWidth, minHeight) FigureCanvas.updateGeometry(self) #第四步:就是畫圖,【可以在此類中畫,也可以在其它類中畫】 #畫日分佈圖 def drawDayChart(self): periodHour=['0'+str(i) if i<10 else str(i) for i in range(24)] periodDict=dict(zip(periodHour, [0]*24)) with open('configFiles/history.txt', 'r') as f: lines=f.readlines() for line in lines: period=line[11:13] periodDict[period]+=1 periodDictKeys=periodDict.keys() periodDictValues=periodDict.values() rects=self.axes.bar(periodDictKeys, periodDictValues, align='edge', width=0.9, color='green') self.axes.set_ylabel('番茄鐘個數') self.axes.set_title('番茄鐘日分佈', color='blue') for rect in rects: x=rect.get_x() y=rect.get_height() if(y==0): continue self.axes.text(x, 1.01*y, str(y), va='bottom')
二、解決思路
1、首先確定每次刷新的時候,穿給圖標的數據是否真正刷新,確定這一點後
2、嘗試使用self.axes.clear()將之前畫的圖刪除掉,然後重畫—嘗試後仍然不行
3、google :How to update a plot in matplotlib?
找到瞭這麼幾行代碼:
fig.canvas.draw() fig.canvas.flush_events()
於是去matplotlib官網查文檔,在我的函數中加入瞭self.fig.canvas.draw_idle()。
4、成功實現瞭刷新
最後的代碼:
class MyFigure(FigureCanvas, QWidget): def __init__(self,parent=None, minWidth=600, minHeight=380, dpi=120): #第一步:創建一個創建Figure self.fig = Figure(figsize=(8, 8), dpi=dpi) #第二步:在父類中**Figure窗口 super(MyFigure,self).__init__(self.fig) #此句必不可少,否則不能顯示圖形 #第三步:創建一個子圖,用於繪制圖形用,111表示子圖編號,如matlab的subplot(1,1,1) self.axes = self.fig.add_subplot(111) self.setMinimumSize(minWidth, minHeight) FigureCanvas.updateGeometry(self) #第四步:就是畫圖,【可以在此類中畫,也可以在其它類中畫】 #畫日分佈圖 def drawDayChart(self): periodHour=['0'+str(i) if i<10 else str(i) for i in range(24)] periodDict=dict(zip(periodHour, [0]*24)) with open('configFiles/history.txt', 'r') as f: lines=f.readlines() for line in lines: period=line[11:13] periodDict[period]+=1 periodDictKeys=periodDict.keys() periodDictValues=periodDict.values() #實現刷新的條件1 self.axes.clear() rects=self.axes.bar(periodDictKeys, periodDictValues, align='edge', width=0.9, color='green') #實現刷新的條件2 self.fig.canvas.draw_idle() self.axes.set_ylabel('番茄鐘個數') self.axes.set_title('番茄鐘日分佈', color='blue') for rect in rects: x=rect.get_x() y=rect.get_height() if(y==0): continue self.axes.text(x, 1.01*y, str(y), va='bottom')
三、原因猜測
即使清空瞭self.axes 但是畫佈中可能還存在殘留數據
self.axes.fig.draw_idle()
在官方文檔中的描述是用於重畫圖表,因此嘗試使用,最終解決瞭問題!
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- matplotlib 向任意位置添加一個子圖(axes)
- PyQt5實現將Matplotlib圖像嵌入到Scoll Area中顯示滾動條效果
- python opencv鼠標畫矩形框之cv2.rectangle()函數
- matplotlib事件處理基礎(事件綁定、事件屬性)
- Python+matplotlib繪制多子圖的方法詳解