Динамическое обновление участка в matplotlib



Я делаю приложение на Python, которое собирает данные из последовательного порта и строит график собранных данных по времени прибытия. Время прибытия данных неизвестно. Я хочу, чтобы сюжет был обновлен при получении данных. Я искал, как это сделать, и нашел два метода:




  1. очистите участок и снова нарисуйте участок со всеми точками.

  2. анимировать сюжет, изменив его после определенного интервала.


Я не предпочитайте первый, так как программа работает и собирает данные в течение длительного времени (например, день), и перерисовка сюжета будет довольно медленной.
Второй также не предпочтителен, так как время прибытия данных неопределенно, и я хочу, чтобы график обновлялся только тогда, когда данные получены.



есть ли способ, которым я могу обновить график, просто добавив к нему больше точек только при получении данных?

724   3  

3 ответов:

есть ли способ, которым я могу обновить сюжет, просто добавив к нему больше точек...

есть несколько способов анимации данных в matplotlib, в зависимости от версии у вас есть. Вы видели matplotlib поваренная книга примеры? Кроме того, проверьте более современный примеры анимации в документации matplotlib. Наконец,анимация API определяет функцию FuncAnimation который анимирует функцию во времени. Эта функция может быть просто функцией, которую вы используете для получения данных.

каждый метод в основном определяет data свойство рисуемого объекта, поэтому не требует очистки экрана или рисунка. Элемент data свойство может быть просто расширено, поэтому вы можете сохранить предыдущие точки и просто продолжать добавлять к своей линии (или изображению или тому, что вы рисуете).

учитывая, что вы говорите, что ваши данные прибытия время неуверенно ваш лучший выбор, вероятно, просто сделать что-то вроде:

import matplotlib.pyplot as plt
import numpy

hl, = plt.plot([], [])

def update_line(hl, new_data):
    hl.set_xdata(numpy.append(hl.get_xdata(), new_data))
    hl.set_ydata(numpy.append(hl.get_ydata(), new_data))
    plt.draw()

затем, когда вы получаете данные из последовательного порта просто позвонить update_line.

для того, чтобы сделать это без FuncAnimation (например, вы хотите, чтобы выполнить другие части кода в то время как сюжет был создан или вы хотите обновить несколько участков одновременно), вызов draw в одиночку не создает график (по крайней мере, с бэкэндом qt).

следующие работы для меня:

import matplotlib.pyplot as plt
plt.ion()
class DynamicUpdate():
    #Suppose we know the x range
    min_x = 0
    max_x = 10

    def on_launch(self):
        #Set up plot
        self.figure, self.ax = plt.subplots()
        self.lines, = self.ax.plot([],[], 'o')
        #Autoscale on unknown axis and known lims on the other
        self.ax.set_autoscaley_on(True)
        self.ax.set_xlim(self.min_x, self.max_x)
        #Other stuff
        self.ax.grid()
        ...

    def on_running(self, xdata, ydata):
        #Update data (with the new _and_ the old points)
        self.lines.set_xdata(xdata)
        self.lines.set_ydata(ydata)
        #Need both of these in order to rescale
        self.ax.relim()
        self.ax.autoscale_view()
        #We need to draw *and* flush
        self.figure.canvas.draw()
        self.figure.canvas.flush_events()

    #Example
    def __call__(self):
        import numpy as np
        import time
        self.on_launch()
        xdata = []
        ydata = []
        for x in np.arange(0,10,0.5):
            xdata.append(x)
            ydata.append(np.exp(-x**2)+10*np.exp(-(x-7)**2))
            self.on_running(xdata, ydata)
            time.sleep(1)
        return xdata, ydata

d = DynamicUpdate()
d()

Я знаю, что опаздываю с ответом на этот вопрос, но для вашего вопроса Вы можете заглянуть в пакет "джойстик". Я разработал его для построения потока данных из последовательного порта, но он работает для любого потока. Это также позволяет для интерактивного ведения журнала текста или графического отображения (в дополнение к графическому отображению). Не нужно делать свои собственные петли в отдельном потоке, пакет заботится об этом, просто дайте частоту обновления, которую вы хотите. Кроме того, терминал остается доступным для команд мониторинга, пока черчение. См.http://www.github.com/ceyzeriat/joystick/ или https://pypi.python.org/pypi/joystick (Используйте Пип установить джойстик для установки)

просто замените np.случайность.random () по вашей реальной точке данных считывается с последовательного порта в коде ниже:

import joystick as jk
import numpy as np
import time

class test(jk.Joystick):
    # initialize the infinite loop decorator
    _infinite_loop = jk.deco_infinite_loop()

    def _init(self, *args, **kwargs):
        """
        Function called at initialization, see the doc
        """
        self._t0 = time.time()  # initialize time
        self.xdata = np.array([self._t0])  # time x-axis
        self.ydata = np.array([0.0])  # fake data y-axis
        # create a graph frame
        self.mygraph = self.add_frame(jk.Graph(name="test", size=(500, 500), pos=(50, 50), fmt="go-", xnpts=10000, xnptsmax=10000, xylim=(None, None, 0, 1)))

    @_infinite_loop(wait_time=0.2)
    def _generate_data(self):  # function looped every 0.2 second to read or produce data
        """
        Loop starting with the simulation start, getting data and
    pushing it to the graph every 0.2 seconds
        """
        # concatenate data on the time x-axis
        self.xdata = jk.core.add_datapoint(self.xdata, time.time(), xnptsmax=self.mygraph.xnptsmax)
        # concatenate data on the fake data y-axis
        self.ydata = jk.core.add_datapoint(self.ydata, np.random.random(), xnptsmax=self.mygraph.xnptsmax)
        self.mygraph.set_xydata(t, self.ydata)

t = test()
t.start()
t.stop()

Comments

    Ничего не найдено.