我有一个python脚本,它读入数据文件并使用matplotlib库显示一个带有四个图的图形。数据文件每隔几秒更新一次,因为它是同时运行的不同软件的输出文件。我希望matplotlib图中的四个图表每隔20秒使用更新的数据文件刷新一次。我实现这个的方式如下:
import pylab as pl
import time
pl.ion()
fig = pl.figure()
while True:
f = open('data.out', 'rb')
#code to parse data and plot four charts
ax = fig.add_subplot(2,2,1)
#...
ax = fig.add_subplot(2,2,4)
#...
pl.draw()
time.sleep(20)
这样可行,但是我失去了缩放和平移按钮的功能,如果调用了pl.show(),它通常会起作用。这不是最佳选择。但是,如果用pl.show()替换pl.draw(),则脚本不再更新绘图。有没有办法动态更新绘图而不会完全失去缩放/平移功能?
答案 0 :(得分:0)
你的代码有点太模糊,无法知道发生了什么。
我可以提供这个: 如果创建子图一次,保存所有轴对象,然后调用show(),则应保留正常功能。
对这些子图的后续更改可以这样完成:
#inside while loop
for i in #subplotlist
ax[i].clear() #ax[i] is the axis object of the i'th subplot
ax[i].plot(#plotstuff)
ax[i].draw()
如果您愿意,可以手动添加缩放和平移工具栏。
答案 1 :(得分:0)
在开发软件时,我想您可能有一种多线程方法。 因此,在这种情况下,使用无限while循环是一个坏主意,就像您要按住主线程一样。
此外,对于GUI来说,突然干扰GUI内部线程(例如wxPython)也是一个坏主意,并且您应该采用事件驱动的设计方法,以免突然中断其他线程(这将导致您的应用程序崩溃)。
使用计时器即可完成工作。
计时器将在以下脚本中执行以下操作:
1 /调用一个函数清除以前的艺术家
2 /重新绘制数据
3 /将更改应用于画布
4 /以以下设计方式创建另一个相同的计时器:一个在完成工作后调用另一个相同计时器的计时器
就像我无法访问您的数据一样,我为插图创建了一个随机数据提供者。 定义的变量delay_repeat允许您在几秒钟内编程刷新。
import pylab as pl
import random
from threading import Timer
def dataprovider():
return [random.randint(0, 8) for i in range(8)]
def random_col():
return ['blue', 'red', 'green', 'orange'][random.randint(0,3)]
# .... #
fig = pl.figure()
axes = [fig.add_subplot(2,2,i) for i in range(1,5)]
paths = [ax.scatter(x=dataprovider(), y=dataprovider(), marker = '+', c=random_col()) for ax in axes]
# .... #
def clear_arts(paths, all_arts=-1):
if all_arts < 0:
all_arts = len(paths)
for path in paths[:all_arts]:
path.remove()
def refresh_arts(paths, delay_repeat):
# 1 - clear previous artists
clear_arts(paths,all_arts=-1)
# 2 - Get artists paths for cleaning
paths = [ax.scatter(x=dataprovider(), y=dataprovider(), marker = '+', c=random_col()) for ax in axes]
# 3 - Apply changes
fig.canvas.draw_idle()
# 4 - Create another timer
Timer(delay_repeat, refresh_arts, (paths, delay_repeat)).start()
# 4- Create a timer that will run function with arguments args and keyword arguments kwargs,
# after interval seconds have passed.
delay_repeat = 2
Timer(delay_repeat, refresh_arts, (paths, delay_repeat)).start()
# print("process continues here")
pl.show()
答案 2 :(得分:0)
您可以这样做。它接受x,y作为列表,并在同一图上输出散点图和线性趋势。
from IPython.display import clear_output
from matplotlib import pyplot as plt
%matplotlib inline
def live_plot(x, y, figsize=(7,5), title=''):
clear_output(wait=True)
plt.figure(figsize=figsize)
plt.xlim(0, training_steps)
plt.ylim(0, 100)
x= [float(i) for i in x]
y= [float(i) for i in y]
if len(x) > 1:
plt.scatter(x,y, label='axis y', color='k')
m, b = np.polyfit(x, y, 1)
plt.plot(x, [x * m for x in x] + b)
plt.title(title)
plt.grid(True)
plt.xlabel('axis x')
plt.ylabel('axis y')
plt.show();