当数据文件发生变化时,如何动态更新matplotlib图?

时间:2011-12-06 20:28:26

标签: python plot matplotlib

我有一个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(),则脚本不再更新绘图。有没有办法动态更新绘图而不会完全失去缩放/平移功能?

3 个答案:

答案 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();

您只需要在循环内调用live_plot(x, y)。外观如下: enter image description here