做个医院网站多少钱,网站h1,网站插件代码大全,网站跟域名备案数量如何在Python Tkinter程序中处理窗口关闭事件(用户单击 X按钮)#xff1f;Tkinter支持一种称为协议处理程序的机制。在这里#xff0c;术语协议是指应用程序和窗口管理器之间的交互。最常用的协议称为WM_DELETE_WINDOW#xff0c;用于定义当用户使用窗口管理器显…如何在Python Tkinter程序中处理窗口关闭事件(用户单击 X按钮)Tkinter支持一种称为协议处理程序的机制。在这里术语协议是指应用程序和窗口管理器之间的交互。最常用的协议称为WM_DELETE_WINDOW用于定义当用户使用窗口管理器显式关闭窗口时发生的情况。您可以使用protocol方法为此协议安装处理程序(小部件必须是Tk或Toplevel小部件)这里有一个具体的例子import tkinter as tkfrom tkinter import messageboxroot tk.Tk()def on_closing():if messagebox.askokcancel(Quit,Do you want to quit?):root.destroy()root.protocol(WM_DELETE_WINDOW, on_closing)root.mainloop()我使用了类似的代码但root.destroy()如果您使用的是像Twisted这样的东西来独立维护事件循环或Tkinter(例如twists反应堆对象)请确保外部主循环已停止并且为此目的提供了任何专门的技术(例如twisted.reactor.stop())在Windows上的Python 2.7上Tkinter没有子模块消息框。 我使用了import tkMessageBox as messagebox我认为您应该知道是您从其他人/其他地方复制了此答案和代码。我不知道那不是我最初发布的代码。不为我工作。 硬关闭窗口时(例如使用Alt F4)它不会改变经典的Python对图形中断的混乱反应。MitchMcMabers另一方面很长一段时间以来tkinter几乎没有任何重大变化。马特展示了关闭按钮的一种经典修改。另一种是使关闭按钮最小化窗口。您可以通过使用iconify方法来重现此行为是协议方法的第二个参数。这是一个在Windows 7上测试过的有效示例# Python 3import tkinterimport tkinter.scrolledtext as scrolledtextclass GUI(object):def __init__(self):root self.root tkinter.Tk()root.title(Test)# make the top right close button minimize (iconify) the main windowroot.protocol(WM_DELETE_WINDOW, root.iconify)# make Esc exit the programroot.bind(, lambda e: root.destroy())# create a menu bar with an Exit commandmenubar tkinter.Menu(root)filemenu tkinter.Menu(menubar, tearoff0)filemenu.add_command(labelExit, commandroot.destroy)menubar.add_cascade(labelFile, menufilemenu)root.config(menumenubar)# create a Text widget with a Scrollbar attachedtxt scrolledtext.ScrolledText(root, undoTrue)txt[font] (consolas, 12)txt.pack(expandTrue, fillboth)gui GUI()gui.root.mainloop()在此示例中我们为用户提供了两个新的退出选项经典文件菜单-退出还有Esc按钮。取决于Tkinter活动尤其是在使用Tkinter.after时使用destroy()停止此活动-即使使用protocol()按钮等-也会干扰此活动(执行时出错)而不是只是终止它。在几乎每种情况下最好的解决方案是使用标志。这是一个简单愚蠢的用法示例(尽管我敢肯定你们大多数人都不需要它from Tkinter import *def close_window():global runningrunning FalseprintWindow closedroot Tk()root.protocol(WM_DELETE_WINDOW, close_window)cv Canvas(root, width200, height200)cv.pack()running True;# This is an endless loop stopped only by setting running to Falsewhile running:for i in range(200):if not running:breakcv.create_oval(i, i, i1, i1)root.update()这样可以很好地终止图形活动。您只需要在正确的位置检查running。尝试简单版本import tkinterwindow Tk()closebutton Button(window, textX, commandwindow.destroy)closebutton.pack()window.mainloop()或者如果您想添加更多命令import tkinterwindow Tk()def close():window.destroy()#More Functionsclosebutton Button(window, textX, commandclose)closebutton.pack()window.mainloop()我要感谢Apostolos的回答这一点引起了我的注意。这是2019年Python 3的更加详细的示例带有更清晰的描述和示例代码。注意以下事实destroy()(或根本没有自定义窗口关闭处理程序)会在用户关闭窗口时立即破坏该窗口及其所有正在运行的回调。这可能对您不利这取决于您当前的Tkinter活动尤其是在使用tkinter.after(定期回调)时。您可能正在使用处理一些数据并将其写入磁盘的回调...在这种情况下您显然希望数据写入完成而不会被突然终止。最好的解决方案是使用标志。因此当用户请求关闭窗口时可以将其标记为一个标志然后对其进行响应。(注意我通常将GUI设计为封装良好的类和单独的工作线程并且我绝对不使用全局(我使用类实例变量)但这只是一个简单的示例用于演示当用户关闭窗口时Tk如何突然终止您的定期回调...)from tkinter import *import time# Try setting this to False and look at the printed numbers (1 to 10)# during the work-loop, if you close the window while the periodic_call# worker is busy working (printing). It will abruptly end the numbers,# and kill the periodic callback! Thats why you should design most# applications with a safe closing callback as described in this demo.safe_closing True# ---------busy_processing Falseclose_requested Falsedef close_window():global close_requestedclose_requested Trueprint(User requested close at:, time.time(),Was busy processing:, busy_processing)root Tk()if safe_closing:root.protocol(WM_DELETE_WINDOW, close_window)lbl Label(root)lbl.pack()def periodic_call():global busy_processingif not close_requested:busy_processing Truefor i in range(10):print((i1),of 10)time.sleep(0.2)lbl[text] str(time.time()) # Will error if force-closed.root.update() # Force redrawing since we change label multiple times in a row.busy_processing Falseroot.after(500, periodic_call)else:print(Destroying GUI at:, time.time())try: #destroy() can throw, so you should wrap it like this.root.destroy()except:# NOTE: In most code, youll wanna force a close here via#exit if the window failed to destroy. Just ensure that# you have no code after your mainloop() call (at the# bottom of this file), since the exit call will cause the# process to terminate immediately without running any more# code. Of course, you should NEVER have code after your# mainloop() call in well-designed code anyway...# exit(0)passroot.after_idle(periodic_call)root.mainloop()该代码将向您显示WM_DELETE_WINDOW处理程序即使在我们的自定义periodic_call()忙于工作/循环的过程中仍在运行我们使用一些非常夸张的.after()值500毫秒。这只是为了使您很容易看到定期呼叫忙时关闭与否之间的区别...如果在数字更新时关闭您将看到WM_DELETE_WINDOW在定期呼叫时发生呼叫正在处理True。如果您在数字暂停时关闭(这意味着此时不处理定期回调)则会看到关闭发生在不忙期间。在实际使用中您的.after()将使用30到100毫秒之类的时间来具有响应GUI。这只是一个演示可以帮助您了解如何保护自己免受Tk的默认关闭时立即中断所有工作的行为。总结使WM_DELETE_WINDOW处理程序设置一个标志然后在安全的情况下(当您的应用程序完成所有工作时)定期并手动检查该标志.destroy()窗口。PS您也可以使用WM_DELETE_WINDOW询问用户是否真的要关闭窗口如果他们回答否则无需设置标志。非常简单您只需在WM_DELETE_WINDOW中显示一个消息框然后根据用户的答案设置标志即可。使用closeEventdef closeEvent(self, event):# code to be executed这个答案需要更多细节。 这条线放在哪里 在我这边摆弄一点似乎无法使它正常工作。