>  기사  >  백엔드 개발  >  pyplot을 이용한 실시간 플로팅

pyplot을 이용한 실시간 플로팅

王林
王林원래의
2024-09-03 17:17:02662검색

Real-time plotting with pyplot

간단한 투표 앱에서 생성한 데이터를 그래프로 표시하고 싶었습니다. 나는 과거에 pyplot을 만지작거렸지만 처음부터 아무것도 만들려고 시도한 적이 없습니다. 운 좋게도 매우 인기가 있으며 StackOverflow 및 다른 곳에서 수많은 예제를 찾을 수 있습니다.

시간이 지남에 따라 그래프 업데이트와 관련된 SO 답변으로 검색을 시작했습니다.

import matplotlib.pyplot as plt
import numpy as np

# You probably won't need this if you're embedding things in a tkinter plot...
plt.ion()

x = np.linspace(0, 6*np.pi, 100)
y = np.sin(x)

fig = plt.figure()
ax = fig.add_subplot(111)
line1, = ax.plot(x, y, 'r-') # Returns a tuple of line objects, thus the comma

for phase in np.linspace(0, 10*np.pi, 500):
    line1.set_ydata(np.sin(x + phase))
    fig.canvas.draw()
    fig.canvas.flush_events()

이 코드는 사인파 변화 단계에 애니메이션을 적용합니다.

처음 두 줄은 사용하려는 라이브러리를 가져옵니다. matplotlib.pyplot은 GUI를 플로팅하고 처리합니다.

이해하면(모르겠지만) ion() 메서드는 pyplot이 GUI를 구동하도록 만듭니다. tkinter 프로그램 내에서 사용할 수도 있고 정적 이미지를 생성하는 데 사용할 수도 있지만, 우리의 경우 플롯의 GUI를 처리하도록 하는 것이 합리적입니다. (나중에 플러시_events() 호출이 수행하는 작업은 Figure 창과의 상호 작용을 허용하는 것입니다.)

이 예에서는 numpy 메서드 linspace()를 사용하여 x 값을 생성합니다. 멋진 Python 목록인 numpy 배열을 반환합니다.

math.sin 대신 np.sin을 사용하는 이유는 방송입니다. 이는 목록의 모든 항목에 함수를 적용하는 numpy 용어입니다. 사실, map을 사용하면 numpy 없이도 동일한 결과를 얻을 수 있다는 생각이 듭니다.

map(lambda n: math.sin(n), x)

하지만 numpy 방송은 사용이 간편하고 간편합니다.

이제 pyplot 설정이 완료되었습니다. 먼저 새로운 "그림"(그림)을 만듭니다. 이 그림에 서브플롯(ax)을 추가하세요. 많이 있을 수 있습니다. 111은 "1x1 그리드를 생성하고 이 서브플롯을 첫 번째 셀에 배치"라는 다소 난해한 해석을 가지고 있습니다.

이 서브플롯(또는 축 집합)에는 전달된 x 및 y 값을 사용하여 선이 그려집니다. (점은 직선으로 연결되어 연속적으로 그려집니다.) "r-"는 빨간색 실선을 지정하는 약식 방법입니다. 여러 줄을 지정할 수 있으므로 플롯()은 튜플을 반환합니다. 위의 코드는 튜플 압축 풀기를 사용하여 원하는 값 하나를 추출합니다.

시작은 좋지만 시간이 지남에 따라 x축을 확장해야 합니다. 또한 이 코드는 필요한 경우 y축의 경계를 업데이트하지 않습니다. 첫 번째 플롯에 대해 계산하는 경계에 고정되어 있습니다. 좀 더 검색하면 이 SO 답변으로 연결됩니다. 인용하자면:

축의 dataLim을 업데이트한 다음 이후에 dataLim을 기반으로 축의 viewLim을 업데이트해야 합니다. 적절한 메서드는 axis.relim() 및 ax.autoscale_view() 메서드입니다.

물론 좋습니다. 그들의 예를 바탕으로 x와 y 모두에서 성장하는 데모 그래프를 만들었습니다.

import matplotlib.pyplot as plt
import numpy as np
from threading import Thread
from time import sleep

x = list(map(lambda x: x / 10, range(-100, 100)))
x_next_max = 100
y = np.sin(x)

# You probably won't need this if you're embedding things in a tkinter plot...
plt.ion()

fig = plt.figure()
ax = fig.add_subplot(111)
line1 = ax.plot(x, y, 'r-')[0] # Returns a tuple of line objects

growth = 0

while True:
    x.append(x_next_max / 10)
    x_next_max += 1
    line1.set_xdata(x)
    line1.set_ydata(np.sin(x) + np.sin(np.divide(x, 100)) + np.divide(x, 100))
    ax.relim()
    ax.autoscale()
    fig.canvas.draw()
    fig.canvas.flush_events()

    sleep(0.1)

이제 어딘가로 가고 있어요. 하지만 이는 차단 루프이므로 데이터를 가끔 업데이트해야 합니다. 스레드가 여러 개인 경우 변수를 업데이트할 때 스레드로부터 안전한지 걱정해야 합니다. 이 경우에는 변수가 5분마다 한 번만 업데이트된다는 것을 알고 있기 때문에(또는 폴링 기능이 자주 실행되는 경우) 게을러질 수 있습니다. 코드 줄 중간에 변수를 덮어쓸 위험이 없습니다.

import matplotlib.pyplot as plt
import numpy as np
from threading import Timer
from time import sleep

x = list(map(lambda x: x / 10, range(-100, 100)))
x_next_max = 100
y = np.sin(x)

# You probably won't need this if you're embedding things in a tkinter plot...
plt.ion()

fig = plt.figure()
ax = fig.add_subplot(111)
line1 = ax.plot(x, y, 'r-')[0] # Plot returns a tuple of line objects

growth = 0
new_x = None

dT = 1

def grow():
    global new_x, x_next_max
    while True:
        new_x = x + [x_next_max / 10]
        x_next_max += 1
        sleep(dT) # grow every dT seconds

t = Thread(target=grow)
t.start()

while True:

    if new_x:
        x = new_x
        new_x = None
        line1.set_xdata(x)
        line1.set_ydata(np.sin(x) + np.sin(np.divide(x, 100)) + np.divide(x, 100))
        ax.relim()
        ax.autoscale()
        fig.canvas.draw()

    fig.canvas.flush_events()

    sleep(0.1)

그래프는 성장 스레드가 new_x에 값을 할당할 때만 업데이트됩니다. flash_events() 호출은 "if" 문 외부에 있으므로 자주 호출된다는 점에 유의하세요.

위 내용은 pyplot을 이용한 실시간 플로팅의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.