matplotlib show () не работает дважды - PullRequest
17 голосов
/ 02 апреля 2011

У меня странная проблема с matplotlib.Если я запускаю эту программу, я могу несколько раз открывать и закрывать одну и ту же фигуру.

import numpy
from pylab import figure, show


X = numpy.random.rand(100, 1000)
xs = numpy.mean(X, axis=1)
ys = numpy.std(X, axis=1)

fig = figure()
ax = fig.add_subplot(111)
ax.set_title('click on point to plot time series')
line, = ax.plot(xs, ys, 'o', picker=5)  # 5 points tolerance


def onpick(event):

    figi = figure()
    ax = figi.add_subplot(111)
    ax.plot([1,2,3,4])        
    figi.show()

fig.canvas.mpl_connect('pick_event', onpick)

show()

Наоборот, если я использую тот же код функции onpick в своем настраиваемом виджете, он открывает фигурутолько в первый раз, в другие события, он входит в функции, но не отображает рисунок:

from PyQt4 import QtGui, QtCore
from matplotlib.backends.backend_qt4agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
import numpy as np
import matplotlib.pyplot as plt
from matplotlib.backends.backend_qt4 import NavigationToolbar2QT as NavigationToolbar
import time

STEP = 0.000152 

class MplCanvas(FigureCanvas):

    def __init__(self):

        # initialization of the canvas
        FigureCanvas.__init__(self, Figure())

        self.queue = []
        self.I_data = np.array([])
        self.T_data = np.array([])

        self.LvsT = self.figure.add_subplot(111)
        self.LvsT.set_xlabel('Time, s')
        self.LvsT.set_ylabel('PMT Voltage, V')
        self.LvsT.set_title("Light vs Time")
        self.LvsT.grid(True)

        self.old_size = self.LvsT.bbox.width, self.LvsT.bbox.height
        self.LvsT_background = self.copy_from_bbox(self.LvsT.bbox)

        self.LvsT_plot, = self.LvsT.plot(self.T_data,self.I_data)
        #self.LvsT_plot2, = self.LvsT.plot(self.T_data2,self.I_data2) 

        self.mpl_connect('axes_enter_event', self.enter_axes)
        self.mpl_connect('button_press_event', self.onpick)
        self.count = 0
        self.draw()

    def enter_axes(self,event):

        print "dentro"

    def onpick(self,event):
        print "click"
        print 'you pressed', event.canvas

        a = np.arange(10)
        print a
        print self.count

        fig = plt.figure()
        ax = fig.add_subplot(111)
        ax.plot(a)    
        fig.show()



    def Start_Plot(self,q,Vmin,Vmax,ScanRate,Cycles):
        self.queue = q

        self.LvsT.clear()
        self.LvsT.set_xlim(0,abs(Vmin-Vmax)/ScanRate*Cycles)
        self.LvsT.set_ylim(-3, 3)
        self.LvsT.set_autoscale_on(False)
        self.LvsT.clear()
        self.draw()

        self.T_data = np.array([])
        self.I_data = np.array([])

        # call the update method (to speed-up visualization)
        self.timerEvent(None)
        # start timer, trigger event every 1000 millisecs (=1sec)
        self.timerLvsT = self.startTimer(3)

    def timerEvent(self, evt):

        current_size = self.LvsT.bbox.width, self.LvsT.bbox.height
        if self.old_size != current_size:
            self.old_size = current_size
            self.LvsT.clear()
            self.LvsT.grid()
            self.draw()
            self.LvsT_background = self.copy_from_bbox(self.LvsT.bbox)

        self.restore_region(self.LvsT_background, bbox=self.LvsT.bbox)

        result = self.queue.get()

        if result == 'STOP': 
            self.LvsT.draw_artist(self.LvsT_plot)
            self.killTimer(self.timerLvsT)
            print "Plot finito LvsT"

        else:
            # append new data to the datasets
            self.T_data = np.append(self.T_data,result[0:len(result)/2])
            self.I_data = np.append(self.I_data,result[len(result)/2:len(result)])

            self.LvsT_plot.set_data(self.T_data,self.I_data)#L_data
            #self.LvsT_plot2.set_data(self.T_data2,self.I_data2)#L_data

            self.LvsT.draw_artist(self.LvsT_plot)

            self.blit(self.LvsT.bbox)


class LvsT_MplWidget(QtGui.QWidget):
    def __init__(self, parent = None):
        QtGui.QWidget.__init__(self, parent)        
        self.canvas = MplCanvas()
        self.vbl = QtGui.QVBoxLayout()
        self.vbl.addWidget(self.canvas)
        self.setLayout(self.vbl)

Этот виджет необходим для анимационного сюжета, и когда эксперимент завершится, если я нажму наНа графике должна появиться фигура, которая появляется только в первый раз.

У вас есть какие-либо подсказки?

Большое спасибо.

Ответы [ 6 ]

9 голосов
/ 03 апреля 2011

У меня есть новая информация о том, что поиск Google появился

Это от автора matplotlib. Это пришло от http://old.nabble.com/calling-show%28%29-twice-in-a-row-td24276907.html

Привет Ондрей,

Я не уверен, где найти хороший объяснение этого, но позвольте мне дать вам некоторые намеки. Предназначен для использования показывать только один раз для каждой программы. а именно «show» должно быть последней строкой в ​​вашем скрипт. Если вы хотите интерактивный заговор вы можете считать интерактивным режим (pyplot.ion-ioff) как в пример ниже.

Кроме того, для динамического построения всех Демо анимации могут быть полезны.

Может быть, вы хотите посмотреть на http://matplotlib.sourceforge.net/users/shell.html .

С наилучшими пожеланиями, Матиас

Так что, похоже, это недокументированная «фича» (ошибка?).

Редактировать: вот его блок кода:

from pylab import *

t = linspace(0.0, pi, 100)
x = cos(t)
y = sin(t)

ion()  # turn on interactive mode
figure(0)
subplot(111, autoscale_on=False, xlim=(-1.2, 1.2), ylim=(-.2, 1.2))

point = plot([x[0]], [y[0]], marker='o', mfc='r', ms=3)

for j in arange(len(t)):
    # reset x/y-data of point
    setp(point[0], data=(x[j], y[j]))
    draw() # redraw current figure

ioff() # turn off interactive mode
show()

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

7 голосов
/ 08 июня 2012

В начале вашего кода включите интерактивный режим через plt.ion ()

1 голос
/ 01 июня 2011

У меня была такая же проблема с show (), работающим только в первый раз.Вы все еще на версии 0.99.3 или около того?Я смог решить мою проблему недавно, если вы все еще заинтересованы в изменении поведения show (), попробуйте это:


Я заметил этот абзац под названием несколько вызовов, чтобы показать, поддерживается на что нового на сайте загрузки matplotlib.

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

Это было «что нового» для версии 1.0.1, на момент написания статьи версия в synaptic все еще была на 0.99.3.Я был в состоянии загрузить и собрать из источника v1.0.1.Дополнительные пакеты, которые я также требовал для удовлетворения зависимостей, были libfreetype6-dev tk-dev tk8.5-dev tcl8.5-dev python-gtk2-dev;Ваш пробег может варьироваться.

Теперь, когда у меня есть matplotlib.__version__ == 1.0.1, следующий код работает так, как я ожидал:

from matplotlib import pyplot as p
from scipy import eye
p.imshow(eye(3))
p.show()
print 'a' 
p.imshow(eye(6))
p.show()
print 'b' 
p.imshow(eye(9))
p.show()
print 'c' 
0 голосов
/ 09 сентября 2017

Вы можете создать экземпляр фигуры с помощью:

fig = plt.figure(0)

И рисовать свои вещи, манипулируя этим рис.Вы можете использовать fig.show() в любое время, чтобы показать свою фигуру.

0 голосов
/ 03 апреля 2011
 def onpick(self,event):
        print "click"
        print 'you pressed', event.canvas
        ...
        ax.plot(a)    
        fig.show() # <--- this blocks the entire loop

Попробуйте:

 def onpick(self,event):
        print "click"
        print 'you pressed', event.canvas
        ...
        ax.plot(a)    
        self.draw()
        self.update()
0 голосов
/ 03 апреля 2011

Мой способ обойти эту проблему - никогда не звонить close.

Я уверен, что вы можете контролировать прозрачность виджета в PyQt.Вы можете попробовать управлять видимостью, используя Qt вместо matplotlib.Я уверен, что кто-то еще, кто знает больше о matplotlib, может дать лучший ответ, чем этот: D

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...