Динамическое обновление участка в matplotlib
Я делаю приложение на Python, которое собирает данные из последовательного порта и строит график собранных данных по времени прибытия. Время прибытия данных неизвестно. Я хочу, чтобы сюжет был обновлен при получении данных. Я искал, как это сделать, и нашел два метода:
- очистите участок и снова нарисуйте участок со всеми точками.
- анимировать сюжет, изменив его после определенного интервала.
Я не предпочитайте первый, так как программа работает и собирает данные в течение длительного времени (например, день), и перерисовка сюжета будет довольно медленной.
Второй также не предпочтителен, так как время прибытия данных неопределенно, и я хочу, чтобы график обновлялся только тогда, когда данные получены.
есть ли способ, которым я могу обновить график, просто добавив к нему больше точек только при получении данных?
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