Прошу прощения за название, но лучшего описания проблемы не придумал. У меня есть программа на основе Tkinter, в которой пользователь может создать отчет в формате PDF, состоящий из обзора, за которым следуют некоторые подробные графики. Я знал, что по какой-то причине вся программа завершит работу после завершения отчета в формате PDF, но я только недавно сел, чтобы точно определить, что вызвало это.
Я обнаружил, что строка plt.close
в начальном обзорном графике приводит к закрытию всей программы после написания отчета в формате pdf (это первая часть, которую я не совсем понимаю, если виновата plot.close
, почему весь модуль работает до завершения)? Во-вторых, почему это вообще происходит?
Минимальный пример, который мне удалось создать (с бессмысленными данными для графиков), приведен ниже, где, если строка, которой предшествует # THE CULPRIT
, закомментирована, экземпляр Tk()
остается живым, но если его оставить как есть, экземпляр Tk()
закрывается.
import tkinter as tk
import matplotlib.pyplot as plt
import numpy as np
from matplotlib.backends.backend_pdf import PdfPages
from pathlib import Path
class Pdf(object):
def __init__(self, master):
self.master = master
pdf = PdfPages(Path.cwd() / 'demo.pdf')
self.pdf = pdf
def plot_initial(self):
fig = plt.figure(figsize=(8,6))
fig.add_subplot(111)
mu, sigma = 0, 0.1
s = np.random.normal(mu, sigma, 1000)
count, bins, ignored = plt.hist(s, 30, density=True)
plt.plot(bins, 1/(sigma * np.sqrt(2 * np.pi)) *
np.exp( - (bins - mu)**2 / (2 * sigma**2) ),
linewidth=2, color='r')
plt.title('Overview')
plt.xlabel('X')
plt.ylabel('Y')
self.pdf.savefig(fig)
# THE CULPRIT
plt.close(fig)
def plot_extra(self):
fig = plt.figure(figsize=(8,6))
fig.add_subplot(111)
mu, sigma = 0, 0.1
s = np.random.normal(mu, sigma, 1000)
count, bins, ignored = plt.hist(s, 30, density=True)
plt.plot(bins, 1/(sigma * np.sqrt(2 * np.pi)) *
np.exp( - (bins - mu)**2 / (2 * sigma**2) ),
linewidth=2, color='r')
plt.title('Extra')
plt.xlabel('X')
plt.ylabel('Y')
self.pdf.savefig(fig)
plt.close(fig)
def close(self):
self.pdf.close()
class MVE(object):
@classmethod
def run(cls):
root = tk.Tk()
MVE(root)
root.mainloop()
def __init__(self, master):
self.root = master
tk.Frame(master)
menu = tk.Menu(master)
master.config(menu=menu)
test_menu = tk.Menu(menu, tearoff=0)
menu.add_cascade(label='Bug', menu=test_menu)
test_menu.add_command(label='PDF', command=
self.generate_pdf)
def generate_pdf(self):
pdf = Pdf(self)
pdf.plot_initial()
for i in range(0,3):
pdf.plot_extra()
pdf.close()
if __name__ == "__main__":
MVE.run()
Версии установленных пакетов/базы Python:
- Питон 3.7.0
- Ткинтер 8.6
- Матплотлиб 2.2.3
- Нампи 1.15.1
Изменить
Я обновился до Matplotlib 3.0.2 в соответствии с предложением @ImportanceOfBeingErnest, однако проблема все еще остается.
plt.close()
делает что-то, что завершает сеансTk()
. Это произойдет только при использовании бэкэнда Tk. Поэтому, если использовать бэкэнд Qt5Agg для pyplot, но оставить приложение tk как есть, вероятно, сработает, но в целом это нежелательно. - person ImportanceOfBeingErnest   schedule 19.02.2019plt.close
. - person Bas Jansen   schedule 19.02.2019"Agg"
, который не требует дополнительного набора инструментов QUI. - person ImportanceOfBeingErnest   schedule 19.02.2019