Tkinter语言 - 运行时错误:超出最大递归深度



我周一开始用Python编程。 我喜欢学习它。 但是我一直在试图理解如何在 tkinter 菜单之间切换时避免递归! 我相信这是一个非常基本的问题,我感谢你容忍我对这个问题的无知,但我无法在其他地方找到答案。

我现在所做的是,最终,给我一个错误:运行时错误:调用 Python 对象时超出了最大递归深度

这是我目前使用的模式。 更新:下面的代码现在是一个完整的、独立的副本,重现了我面临的问题!:D

from tkinter import *
def mainmenu():
    global frame, root
    frame.destroy()
    frame = Frame()
    frame.pack()
    button1 = Button(frame, text="anothermenulikethis", command = anothermenulikethis)
    button2 = Button(frame, text="anothermenulikethis", command = anothermenulikethis)
    button3 = Button(frame, text="mainmenu", command = mainmenu)
    button1.pack(side=LEFT)
    button2.pack(side=LEFT)
    button3.pack(side=LEFT)
    root.mainloop()
def anothermenulikethis():
    global frame, root
    frame.destroy()
    frame = Frame()
    frame.pack()
    button1 = Button(frame, text="mainmenu", command = mainmenu)
    button2 = Button(frame, text="mainmenu", command = mainmenu)
    button3 = Button(frame, text="anothermenulikethis", command = anothermenulikethis)
    button1.pack(side=LEFT)
    button2.pack(side=LEFT)
    button3.pack(side=LEFT)
    root.mainloop()
root = Tk()
root.title("Recursive Menu Problem Isolation")
root.geometry("1200x600")
frame = Frame()
mainmenu()

一切都工作正常,直到它不可避免地从最大递归深度失败。 如果有人能提出更好的做事方法,或者有一个更好的方法的例子的链接,我渴望学习。

PS:我已经查看并尝试增加递归深度,但我觉得这是一个穷人对我的方法的根本问题的解决方案。

提前谢谢大家。

根据要求,以下是回溯:

Exception in Tkinter callback
Traceback (most recent call last):
  File "/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/tkinter/__init__.py", line 1399, in __call__
    return self.func(*args)
  File "/Users/diligentstudent/Desktop/menutest.py", line 11, in mainmenu
    button1 = Button(frame, text="anothermenulikethis", command = anothermenulikethis)
  File "/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/tkinter/__init__.py", line 2028, in __init__
    Widget.__init__(self, master, 'button', cnf, kw)
  File "/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/tkinter/__init__.py", line 1958, in __init__
    (widgetName, self._w) + extra + self._options(cnf))
  File "/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/tkinter/__init__.py", line 1043, in _options
    v = self._register(v)
  File "/Library/Frameworks/Python.framework/Versions/3.2/lib/python3.2/tkinter/__init__.py", line 1079, in _register
    f = CallWrapper(func, subst, self).__call__
RuntimeError: maximum recursion depth exceeded
只需要

一个mainloop()来处理tkinter GUI。

话虽如此,我认为您只需要一个类结构的示例:

from tkinter import Tk,Button
class Application(Tk):
    def say_hi(self):
        print('Hello world?!')
    def close_app(self):
        self.destroy()
    def create_Widgets(self):
        self.quitButton = Button(self, width=12, text='Quit', bg='tan',
                    command=self.close_app)
        self.quitButton.grid(row=0, column=0, padx=8, pady=8)
        self.helloButton = Button(self, width=12, text='Hello',
                    command=self.say_hi)
        self.helloButton.grid(row=0, column=1, padx=8, pady=8)
    def __init__(self):
        Tk.__init__(self)
        self.title('Hello world!')
        self.create_Widgets()
app = Application()
app.mainloop()

为了避免与其他模块可能发生冲突,有些人更喜欢像这样
导入(清楚地说明一切来自哪里):

import tkinter as tk
class Application(tk.Tk):
    def __init__(self):
        tk.Tk.__init__(self)
        self.title('Hello world!')
        self.quitButton = tk.Button(self, width=12, text='Quit', bg='tan',
                    command=self.close_app)
        self.quitButton.grid(row=0, column=0, padx=8, pady=8)
        self.helloButton = tk.Button(self, width=12, text='Hello',
                    command=self.say_hi)
        self.helloButton.grid(row=0, column=1, padx=8, pady=8)
    def say_hi(self):
        print('Hello world?!')
    def close_app(self):
        self.destroy()
app = Application()
app.mainloop()

如您所见,创建小部件很容易在__init__


我决定根据过去一个月学到的知识做一个更实用/有教育意义的例子。在这样做的时候,我得到了一点启示:不是所有的事情都需要自我。类中的前缀!对于 tkinter 类尤其如此,因为您不会将其作为主程序中的对象进行操作。大多数情况下,你需要自我。稍后要在方法中使用某些内容时的前缀。前面的示例显示了任何内容(如按钮)如何接收自我。前缀,即使完全没有必要。

此示例将显示的一些内容:

pack()grid() 可以在同一 GUI 中使用,只要它们不共享主节点。

• 文本小部件可以在字体大小更改时不展开。

• 如何打开和关闭所选文本的粗体标签。

• 如何真正使 GUI 在屏幕上居中。(更多信息在这里)

• 如何使顶级窗口相对于主窗口显示在相同的位置。

• 防止顶级窗口被破坏的两种方法,因此只需创建一次。

• 使 ctrl+a(全选)功能正常。

import tkinter as tk
import tkFont
class Application(tk.Tk):
    def __init__(self):
        tk.Tk.__init__(self)
        self.title('T-Pad')
    # Menubar
        menubar = tk.Menu(self)
        filemenu = tk.Menu(menubar, tearoff=0)
        filemenu.add_command(label="Exit", command=self.close_app)
        menubar.add_cascade(label="File", menu=filemenu)
        formatmenu = tk.Menu(menubar, tearoff=0)
        formatmenu.add_command(label="Font", command=self.show_sizeWin)
        menubar.add_cascade(label="Format", menu=formatmenu)
        self.config(menu=menubar)
    # Bold Button
        boldButton = tk.Button(self, width=12, text='Bold',
                                command=self.make_bold)
        boldButton.pack()
    # Text widget, its font and frame
        self.defaultFont = tkFont.Font(name="defFont")
        textFrame = tk.Frame(self, borderwidth=1, relief="sunken",
                             width=600, height=600)
        textFrame.grid_propagate(False) # ensures a consistent GUI size
        textFrame.pack(side="bottom", fill="both", expand=True)

        self.mText = tk.Text(textFrame, width=48, height=24, wrap='word',
                            font="defFont")
        self.mText.grid(row=0, column=0, sticky="nsew")
    # Scrollbar and config
        tScrollbar = tk.Scrollbar(textFrame, command=self.mText.yview)
        tScrollbar.grid(row=0, column=1, sticky='nsew', pady=1)
        self.mText.config(yscrollcommand=tScrollbar.set)
    # Stretchable
        textFrame.grid_rowconfigure(0, weight=1)
        textFrame.grid_columnconfigure(0, weight=1)
    # Bold Tag
        self.bold_font = tkFont.Font(self.mText, self.mText.cget("font"))
        self.bold_font.configure(weight="bold")
        self.mText.tag_configure("bt", font=self.bold_font)
    # Center main window
        self.update_idletasks()
        xp = (self.winfo_screenwidth() / 2) - (self.winfo_width() / 2) - 8
        yp = (self.winfo_screenheight() / 2) - (self.winfo_height() / 2) - 30
        self.geometry('{0}x{1}+{2}+{3}'.format(self.winfo_width(), self.winfo_height(),
                                                                                xp, yp))
    # Font Size Window (notice that self.sizeWin is given an alias)
        sizeWin = self.sizeWin = tk.Toplevel(self, bd=4, relief='ridge')
        self.sizeList = tk.Listbox(sizeWin, width=10, height=17, bd=4,
                                font=("Times", "16"), relief='sunken')
        self.sizeList.grid()
        doneButton = tk.Button(sizeWin, text='Done', command=sizeWin.withdraw)
        doneButton.grid()
        for num in range(8,25):
            self.sizeList.insert('end', num)
        sizeWin.withdraw()
        sizeWin.overrideredirect(True) # No outerframe!
        # Below is another way to prevent a TopLevel window from being destroyed.
        # sizeWin.protocol("WM_DELETE_WINDOW", self.callback)
    # Bindings
        # Double click a font size in the Listbox
        self.sizeList.bind("<Double-Button-1>", self.choose_size)
        self.bind_class("Text", "<Control-a>", self.select_all)
##    def callback(self):
##        self.sizeWin.withdraw()
    def select_all(self, event):
        self.mText.tag_add("sel","1.0","end-1c")
    def choose_size(self, event=None):
        size_retrieved = self.sizeList.get('active')
        self.defaultFont.configure(size=size_retrieved)
        self.bold_font.configure(size=size_retrieved)
    def show_sizeWin(self):
        self.sizeWin.deiconify()
        xpos = self.winfo_rootx() - self.sizeWin.winfo_width() - 8
        ypos = self.winfo_rooty()
        self.sizeWin.geometry('{0}x{1}+{2}+{3}'.format(self.sizeWin.winfo_width(),
                                                self.sizeWin.winfo_height(), xpos, ypos))
    def make_bold(self):
        try:
            current_tags = self.mText.tag_names("sel.first")
            if "bt" in current_tags:
                self.mText.tag_remove("bt", "sel.first", "sel.last")
            else:
                self.mText.tag_add("bt", "sel.first", "sel.last")
        except tk.TclError:
            pass
    def close_app(self):
        self.destroy()
app = Application()
app.mainloop()

给有此问题的其他人的注意事项:您的按钮命令可能不在正确的缩进级别! 在进一步挖掘之前,请检查它是否与您的其他类方法内联。 不久前我自己遇到了这个问题,重新检查我的缩进解决了一切。

最新更新