我有一个倒计时脚本,当我一次又一次按下开始和暂停按钮时,秒数会快速更新。
起初,我认为在开始按钮和暂停按钮之间快速单击会加快秒速,因此我添加了time.sleep(1)
来延迟在无效按钮之间的单击。
如何解决此问题?
我的算法正确吗?还是还有其他更好的方法来倒计时?如果是,该怎么办?
代码
from tkinter import *
PAUSE = False
HOUR, MINUTE, SECOND = 0, 0, 0
def start():
'''Command for START button'''
global PAUSE
PAUSE = False
start_button['state'] = 'disabled'
pause_button['state'] = 'normal'
reset_button['state'] = 'normal'
# time.sleep(1) to delay time between clicks of pause and start buttons
Counter()
def pause():
'''Command for PAUSE button'''
global PAUSE
PAUSE = True
start_button['state'] = 'normal'
pause_button['state'] = 'disabled'
def reset():
'''Command for RESET button'''
global HOUR, MINUTE, SECOND, PAUSE
PAUSE = True
start_button['state'] = 'normal'
pause_button['state'] = 'disabled'
reset_button['state'] = 'disabled'
Time['text'] = '00:00:00'
HOUR, MINUTE, SECOND = 0, 0, 0
def Counter():
'''Updating hour, minute and seconds'''
global HOUR, MINUTE, SECOND
if PAUSE is False:
if SECOND == 59:
if MINUTE == SECOND == 59:
HOUR += 1
if MINUTE == 59:
MINUTE = 0
else:
MINUTE += 1
SECOND = -1
SECOND += 1
Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
root.after(1000, Counter)
root = Tk()
root.title('COUNTER')
width, height = 342, 108
pos_x = root.winfo_screenwidth() // 2 - width // 2
pos_y = root.winfo_screenheight() // 2 - height // 2
root.geometry('{}x{}+{}+{}'.format(width, height, pos_x, pos_y))
Time = Label(root, fg='Black', text='00:00:00', font=("Helvetica", 40))
Time.pack(side='bottom')
start_button = Button(root, text='START', font=("Arial", 16), fg='black', width=8, command=start)
start_button.pack(side='left')
pause_button = Button(root, text='PAUSE', font=("Arial", 16), fg='black', width=8, state='disabled', command=pause)
pause_button.pack(side='left')
reset_button = Button(root, text='RESET', font=("Arial", 16), fg='black', width=10, state='disabled', command=reset)
reset_button.pack(side='left', fill='both')
root.mainloop()
答案 0 :(得分:1)
您的Counter
函数每次单击都会调用root.after(1000,Counter)
,并在单击后立即修改Time
标签。当您足够快地单击按钮时,您可以安排多个root.after
并累加秒数。
要修改当前脚本,您可以跟踪当前操作,并调用root.after_cancel
暂停操作。
from tkinter import *
PAUSE = False
HOUR, MINUTE, SECOND = 0, 0, 0
action = None #keep track on current action and also avoid click spam
def start():
'''Command for START button'''
global PAUSE, action
PAUSE = False
...
if not action:
Counter()
def pause():
'''Command for PAUSE button'''
global PAUSE, action
...
if action:
root.after_cancel(action)
action = None
def reset():
'''Command for RESET button'''
global HOUR, MINUTE, SECOND, PAUSE, action
...
HOUR, MINUTE, SECOND = 0, 0, 0
action = None
def Counter():
'''Updating hour, minute and seconds'''
global HOUR, MINUTE, SECOND, action
if PAUSE is False:
if SECOND == 59:
if MINUTE == SECOND == 59:
HOUR += 1
if MINUTE == 59:
MINUTE = 0
else:
MINUTE += 1
SECOND = -1
SECOND += 1
Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
action = root.after(1000, Counter)
root = Tk()
...
start_button = Button(root, text='START', font=("Arial", 16), fg='black', width=8, command=lambda: root.after(1000,start)) #to avoid the instant second increment
答案 1 :(得分:1)
起初,我认为快速单击开始按钮和暂停按钮之间的速度会加快秒数,因此我添加了time.sleep(1)来延迟无效按钮之间的单击。
问题在于,由于.aftert()
方法调用自身之前有一个延迟,因此多次单击开始按钮并创建.after(1000, counter)
的多个循环。意味着一次发生了许多.after()
循环。要解决此问题,您只需要进行修改,这样就在出现.after()
循环的情况下无法按下开始按钮。
为此,您必须更改三个功能
counter()
函数
我更改了代码,因此如果计数器正在运行,则start_button
将被禁用,并且当计数器停止运行时,其状态将恢复为正常
def Counter():
'''Updating hour, minute and seconds'''
global HOUR, MINUTE, SECOND
if PAUSE is False:
start_button['state'] = 'disabled' # setting it to disabled if the counter is running
if SECOND == 59:
if MINUTE == SECOND == 59:
HOUR += 1
if MINUTE == 59:
MINUTE = 0
else:
MINUTE += 1
SECOND = -1
SECOND += 1
Time.config(text='{}:{}:{}'.format(str(HOUR).zfill(2), str(MINUTE).zfill(2), str(SECOND).zfill(2)))
root.after(1000, Counter)
else:
start_button['state'] = 'normal' # setting it to normal if the counter stops
pause()
和reset()
函数
您必须删除将start_button
的状态设置为正常的功能,因为这将允许在上一次计数器循环结束之前按下start_button
def pause():
'''Command for PAUSE button'''
global PAUSE
PAUSE = True
pause_button['state'] = 'disabled'
def reset():
'''Command for RESET button'''
global HOUR, MINUTE, SECOND, PAUSE
PAUSE = True
pause_button['state'] = 'disabled'
reset_button['state'] = 'disabled'
Time['text'] = '00:00:00'
HOUR, MINUTE, SECOND = 0, 0, 0
更新这些功能将消除此错误。 希望这是您正在寻找的答案:)