1

我做了一个小脚本来练习 tkinter 的使用。我希望程序打开一个窗口并显示一个标签。按下按钮后,标签应显示 0 到 100 之间的随机数。另外我希望标签每秒刷新一次并显示另一个随机数。

from tkinter import *
import random
import time
root = Tk()

def getrandint():
    result = random.randint(0, 100)
    return result


def go():
    lab2['text'] = 'Number: ' + str(getrandint())
    lab2.pack()

    root.geometry('300x200')
    root.mainloop()
    time.sleep(1)
    go()


lab1 = Label(root, text='Type in number')
lab2 = Label(root, text='Number:')
#ent = Entry(root, width=20)
#number = ent.get()
b = Button(root, text='Go', command=go())

b.pack()
lab1.pack()
lab2.pack()
#ent.pack()

这就是我走多远。它会打开一个窗口并显示一个随机数,但不会刷新该数字。按钮甚至没有显示。此外,当我关闭窗口时,Python 3.8 会显示以下错误消息:

Traceback (most recent call last):
  File "C:/Users/chris/Desktop/WeatherLinkPy/testing.py", line 102, in <module>
    b = Button(root, text='Go', command=go())
  File "C:/Users/chris/Desktop/WeatherLinkPy/testing.py", line 95, in go
    go()
  File "C:/Users/chris/Desktop/WeatherLinkPy/testing.py", line 89, in go
    lab2['text'] = 'Number: ' + str(getrandint())
  File "C:\Users\chris\AppData\Local\Programs\Python\Python38\lib\tkinter\__init__.py", line 1660, in __setitem__
    self.configure({key: value})
  File "C:\Users\chris\AppData\Local\Programs\Python\Python38\lib\tkinter\__init__.py", line 1649, in configure
    return self._configure('configure', cnf, kw)
  File "C:\Users\chris\AppData\Local\Programs\Python\Python38\lib\tkinter\__init__.py", line 1639, in _configure
    self.tk.call(_flatten((self._w, cmd)) + self._options(cnf))
_tkinter.TclError: invalid command name ".!label2"

最后,有没有办法改变random.randint(0, b) 的第二个参数,在开头加上一个入口和一个按钮?

4

3 回答 3

1
  • 让我们从基本错误开始,您的位置mainloop()决定了所有小部件的显示内容(在编写良好的代码上)。在这种情况下,你想把它放在mainloop()最后。在您的情况下,这甚至可以工作,因为您正在使用()具有包含功能的按钮mainloop()(通过文字来解释有点棘手:P)。

  • 现在下一个是,您不应该调用()用于按钮的函数,因为它会自动启动该函数并且不会等待按下按钮:

b = Button(root, text='Go', command=go)
  • 接下来是修复您的条目小部件和事件驱动编程。您不应该在代码开始时从用户那里获取输入,因为它在开始时会是空的。在触发事件(一个函数左右)后,您应该获取输入并将其存储在变量中,这意味着您getrandint()将:
def getrandint():
    try: # To ignore if non integers are entered into the entry widget
        result = random.randint(0, int(ent.get())) # Get the text from entry widget
        return result
    except TypeError: # Ignore the error
        pass
  • 接下来是避免使用,time.sleep()因为它会滞后于 GUI。因此,您应该改用root.after(ms,func)which 将func在指定之后调用ms
def go():
    lab2['text'] = 'Number: ' + str(getrandint())
    root.after(1000,go) # Repeat the function after 1 second

所以你的最终代码是:

from tkinter import *
import random

root = Tk()
root.geometry('300x200')

def getrandint():
    try:
        result = random.randint(0, int(ent.get()))
        return result
    except TypeError:
        pass

def go():
    lab2['text'] = 'Number: ' + str(getrandint())
    root.after(1000,go)

lab1 = Label(root, text='Type in number')
lab2 = Label(root, text='Number:')
ent = Entry(root, width=20)
b = Button(root, text='Go', command=go)

b.pack()
lab1.pack()
lab2.pack()
ent.pack()

root.mainloop() # At the end of the code

另请注意:

您可以通过将您getrandint()的更改go()为:

def go():
    try: # If not an integer is entered into entry widget
        lab2['text'] = 'Number: ' + str(random.randint(0,int(ent.get())))
        root.after(1000,go)
    except ValueError: # Ignore the error
        pass

综上所述,您的错误是由于您的代码流动方式而发生的。经常练习牢记这些,您将永远不会再看到这些错误;)

于 2021-01-04T14:04:02.510 回答
0

首先,当您将命令传递给按钮时,您需要将函数放在不带括号的位置:command=go。如果你使用括号,函数将在你声明按钮时调用,而不是在你按下它时调用。

其次,一旦按下按钮,您不应该调用 root.mainloop() 。您应该在文件末尾运行它。

mainloop永远不会返回,因此 putsleep和函数调用 after 无济于事。如果您想在主循环期间运行代码,请创建一个调用tk.update()和的循环tk.update_idletasks()

我不认为更新的元素需要再次打包来更新。

go最后,你为什么要从里面打电话?

from tkinter import *
import random
import time
root = Tk()

def getrandint():
    result = random.randint(0, 100)
    return result


def go():
    lab2['text'] = 'Number: ' + str(getrandint())


lab1 = Label(root, text='Type in number')
lab2 = Label(root, text='Number:')
#ent = Entry(root, width=20)
#number = ent.get()
b = Button(root, text='Go', command=go)

b.pack()
lab1.pack()
lab2.pack()
#ent.pack()

root.geometry('300x200')
root.mainloop()
于 2021-01-04T14:03:35.123 回答
0

根据您的指示进行的程序应如下所示(类似):

from tkinter import *
import random
import time

# Create function, which generates random number and update label
def set_random_num():
    num = random.randint(0,100)
    lab2.config(text=num)

# Initialize tkinter window
root = Tk()

# Create labels and button
lab1 = Label(root, text='Type in number')
lab1.pack()

lab2 = Label(root, text='Number:')
lab2.pack()

b = Button(root, text='Go', command=lambda: set_random_num()) 
b.pack()

# Keep it at the end of program
root.mainloop() 

Lambda函数允许你正确调用你的函数。我建议为 tkinter 寻找一些好的教程。试试这个

于 2021-01-04T14:09:13.667 回答