- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想要一个带有进度条和一些条目、标签和按钮小部件的顶级窗口/对话框。我希望从 main_window 窗口更新对话框。 main_window 完成工作,我需要将其反射(reflect)在对话框中。我希望主窗口保持事件状态,以便您可以停止该过程。我还希望能够在对话框中停止该过程。
如果不使用多处理和线程,我就无法让它工作。似乎我以错误的方式解决这个问题,还是我?另外,我是多处理和线程的新手,所以我希望无论如何我都做对了。如果有人知道更好的方法来做到这一点,请告诉我。
下面是我尝试做我想做的事情,它有效,但这是正确的方法吗?
我的第一次尝试:
import tkinter as tk
import tkinter.ttk as ttk
from time import sleep
from queue import Empty
from threading import Thread
from multiprocessing import Process, Queue
HIDE = -1
STOP = -2
BREAK = -3
PAUSE = -4
RESUME = -5
class App(tk.Tk):
def __init__(self, **kwargs):
title = kwargs.pop('title', '')
theme = kwargs.pop('theme', 'clam')
geometry = kwargs.pop('geometry', None)
exit_callback = kwargs.pop('exit_callback', None)
super().__init__(**kwargs)
self.title(title)
self.style = ttk.Style()
self.style.theme_use(theme)
if geometry:
self.geometry(geometry)
if exit_callback:
self.protocol('WM_DELETE_WINDOW', exit_callback)
def main_window(out_que, in_que, maximum):
def worker():
if app.running:
return
app.running = True
app.finished = False
for count in range(0, maximum + 1):
try:
message = in_que.get_nowait()
if message:
if message == PAUSE:
message = in_que.get()
if message == BREAK:
break
elif message == STOP:
app.destroy()
except Empty:
pass
sleep(0.1) # Simulate work.
out_que.put(count)
app.running = False
app.finished = True
start_btn.config(state=tk.NORMAL)
def app_stop():
out_que.put(STOP)
app.destroy()
def test_stop():
if app.running:
out_que.put(HIDE)
elif app.finished:
out_que.put(HIDE)
in_que.get()
stop_btn.config(state=tk.DISABLED)
start_btn.config(state=tk.NORMAL)
def test_start():
while not in_que.empty():
in_que.get()
stop_btn.config(state=tk.NORMAL)
start_btn.config(state=tk.DISABLED)
thread = Thread(target=worker, daemon=True)
thread.daemon = True
thread.start()
app = App(title='Main Window', theme='alt', geometry='350x150', exit_callback=app_stop)
app.running = False
app.finished = True
app.rowconfigure(0, weight=1)
app.rowconfigure(1, weight=1)
app.columnconfigure(0, weight=1)
start_btn = ttk.Button(app, text='Start Test', command=test_start)
start_btn.grid(padx=10, pady=5, sticky=tk.NSEW)
stop_btn = ttk.Button(app, text='Stop Test', state=tk.DISABLED, command=test_stop)
stop_btn.grid(padx=10, pady=5, sticky=tk.NSEW)
app.mainloop()
def progress_window(in_que, out_que, maximum):
def hide():
out_que.put(BREAK)
pause_btn.config(text='Pause')
app.withdraw()
def pause():
if progress_bar['value'] < progress_bar['maximum']:
text = pause_btn.cget('text')
text = 'Resume' if text == 'Pause' else 'Pause'
pause_btn.config(text=text)
out_que.put(PAUSE)
else:
pause_btn.config(text='Pause')
def worker():
while True:
data = in_que.get()
print(data)
if data == HIDE:
hide()
elif data == STOP:
app.destroy()
out_que.put(STOP)
break
elif not data:
app.deiconify()
progress_bar["value"] = 0
else:
progress_bar["value"] = data
app.update_idletasks()
app = App(title='Progress', theme='clam', geometry='350x150', exit_callback=hide)
app.rowconfigure(0, weight=1)
app.rowconfigure(1, weight=1)
app.columnconfigure(0, weight=1)
app.columnconfigure(1, weight=1)
progress_bar = ttk.Progressbar(app, orient=tk.HORIZONTAL, mode='determinate')
progress_bar["maximum"] = maximum
progress_bar.grid(padx=10, sticky=tk.EW, columnspan=1000)
pause_btn = ttk.Button(app, text='Pause', command=pause)
pause_btn.grid()
cancel_btn = ttk.Button(app, text='Cancel', command=hide)
cancel_btn.grid(row=1, column=1)
thread = Thread(target=worker)
thread.daemon = True
thread.start()
app.withdraw()
app.mainloop()
if __name__ == '__main__':
jobs = []
que1 = Queue()
que2 = Queue()
process = 50 # The maximum amount of work to process, # items.
for target in (main_window, progress_window):
p = Process(target=target, args=(que1, que2, process))
jobs.append(p)
p.start()
for j in jobs:
j.join()
import tkinter as tk
import tkinter.ttk as ttk
from time import sleep
from queue import Empty
from threading import Thread
from multiprocessing import Queue
HIDE = -1
STOP = -2
DONE = -3
BREAK = -4
PAUSE = -5
class App(tk.Tk):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.running = False
self.finished = True
self.app_que = Queue()
self.dialog_que = Queue()
self.process_items = 50
self.rowconfigure(0, weight=1)
self.rowconfigure(1, weight=1)
self.columnconfigure(0, weight=1)
self.title('Main Window')
self.geometry('350x150')
self.style = ttk.Style()
self.style.theme_use('clam')
wdg = self.start_btn = ttk.Button(self, text='Start Test', command=self.test_start)
wdg.grid(padx=10, pady=5, sticky=tk.NSEW)
wdg = self.stop_btn = ttk.Button(self, text='Stop Test', state=tk.DISABLED, command=self.test_stop)
wdg.grid(padx=10, pady=5, sticky=tk.NSEW)
self.dlg = ProgressDialog(self, title='Progress', geometry='350x150', process=self.process_items)
self.dlg.app_que = self.app_que
self.dlg.dialog_que = self.dialog_que
self.protocol('WM_DELETE_WINDOW', self.app_stop)
thread = Thread(target=self.dlg.worker, daemon=True)
thread.start()
def worker(self):
self.dlg.cancel_btn.config(text='Cancel')
self.dlg.pause_btn.config(state=tk.NORMAL)
for count in range(0, self.process_items + 1):
try:
message = self.app_que.get_nowait()
if message:
if message == PAUSE:
message = self.app_que.get()
if message == BREAK:
self.stop_btn.config(state=tk.DISABLED)
break
elif message == STOP:
self.destroy()
except Empty:
pass
sleep(0.1) # Simulate work.
self.dialog_que.put(count)
self.dialog_que.put(DONE)
self.dlg.cancel_btn.config(text='Close')
self.finished = True
self.start_btn.config(state=tk.NORMAL)
self.stop_btn.config(state=tk.DISABLED)
def app_stop(self):
self.dialog_que.put(STOP)
self.destroy()
def test_stop(self):
if self.running or self.finished:
self.dialog_que.put(HIDE)
self.stop_btn.config(state=tk.DISABLED)
self.start_btn.config(state=tk.NORMAL)
def test_start(self):
while not self.app_que.empty():
self.app_que.get()
thread = Thread(target=self.worker, daemon=True)
thread.start()
self.stop_btn.config(state=tk.NORMAL)
self.start_btn.config(state=tk.DISABLED)
self.dlg.deiconify()
class ProgressDialog(tk.Toplevel):
def __init__(self, parent, *args, **kwargs):
title = kwargs.pop('title', '')
process = kwargs.pop('process', 0)
geometry = kwargs.pop('geometry', None)
super().__init__(parent, *args, **kwargs)
self.withdraw()
self.app_que = None
self.dialog_que = None
self.rowconfigure(0, weight=1)
self.rowconfigure(1, weight=1)
self.columnconfigure(0, weight=1)
self.columnconfigure(1, weight=1)
self.title(title)
if geometry:
self.geometry(geometry)
wdg = self.progress_bar = ttk.Progressbar(self, orient=tk.HORIZONTAL, mode='determinate')
wdg["value"] = 0
wdg["maximum"] = process
wdg.grid(padx=10, sticky=tk.EW, columnspan=1000)
wdg = self.pause_btn = ttk.Button(self, text='Pause', command=self.pause)
wdg.grid()
wdg = self.cancel_btn = ttk.Button(self, text='Cancel', command=self.hide)
wdg.grid(row=1, column=1)
self.protocol('WM_DELETE_WINDOW', self.hide)
def worker(self):
while True:
message = self.dialog_que.get()
print(message)
if message == HIDE:
self.hide()
elif message == STOP:
self.app_que.put(DONE)
break
elif message == DONE:
self.pause_btn.config(state=tk.DISABLED)
else:
self.progress_bar["value"] = message
def hide(self):
self.app_que.put(BREAK)
self.pause_btn.config(text='Pause')
self.withdraw()
def pause(self):
if self.progress_bar['value'] < self.progress_bar['maximum']:
text = self.pause_btn.cget('text')
text = 'Resume' if text == 'Pause' else 'Pause'
self.pause_btn.config(text=text)
self.app_que.put(PAUSE)
else:
self.pause_btn.config(text='Pause')
if __name__ == '__main__':
app = App()
app.mainloop()
最佳答案
如果您不想使用 thread
,也许你可以试试 asyncio
.我不知道我的代码是否正确,但它在我的 PC 上运行良好。
欢迎指出我代码中的错误,我真的不知道这是否是一个好的做法。
import tkinter as tk
from tkinter import ttk
import asyncio, time
import warnings
class App(tk.Tk):
def __init__(self):
super(App, self).__init__()
self.start_btn = ttk.Button(self, text="Start Test", command=self.test_start)
self.start_btn.pack(padx=10, pady=5, fill="both", expand=True)
self.stop_btn = ttk.Button(self, text="Stop Test", command=self.test_stop, state=tk.DISABLED)
self.stop_btn.pack(padx=10, pady=5, fill="both", expand=True)
self.test_window = tk.Toplevel()
self.test_window.progressbar = ttk.Progressbar(self.test_window, orient=tk.HORIZONTAL)
self.test_window.progressbar.grid(padx=10, pady=5, sticky=tk.NSEW, columnspan=2, column=0, row=0)
self.test_window.switch_btn = ttk.Button(self.test_window, text="Pause", command=self.switch)
self.test_window.switch_btn.grid(padx=10, pady=5, sticky=tk.NSEW, column=0, row=1)
self.test_window.cancel_btn = ttk.Button(self.test_window, text="Cancel", command=self.test_cancel)
self.test_window.cancel_btn.grid(padx=10, pady=5, sticky=tk.NSEW, column=1, row=1)
self.test_window.withdraw()
def test_start(self):
self.stop_btn['state'] = tk.NORMAL
self.test_window.deiconify()
self.test_window.after(0, self.work)
def work(self):
async def async_work(): # define a async task
try:
await asyncio.sleep(3) # could be another async work.
except asyncio.CancelledError:
print("cancel or stop")
raise # if don't raise the error ,it won't cancel
async def progressbar_add():
self.task = asyncio.create_task(async_work())
timeout = 0
while True: # wait the async task finish
done, pending = await asyncio.wait({self.task}, timeout=timeout)
self.test_window.update()
if self.task in done:
self.test_window.progressbar['value'] += 10 # if finished, value += 10
print(self.test_window.progressbar['value'])
await self.task
break
if self.test_window.progressbar['value'] >= 100:
return
asyncio.run(progressbar_add())
self.test_window.after(0, self.work)
def test_stop(self):
self.test_window.progressbar['value'] = 0
self.stop_btn['state'] = tk.DISABLED
try:
all_tasks = asyncio.Task.all_tasks()
for task in all_tasks:
task.cancel()
except RuntimeError: # if you have cancel the task it will raise RuntimeError
pass
def switch(self):
if self.test_window.switch_btn['text'] == 'Pause':
self.test_window.switch_btn['text'] = 'Resume'
try:
all_tasks = asyncio.Task.all_tasks()
for task in all_tasks:
task.cancel()
except RuntimeError: # if you have cancel the task it will raise RuntimeError
pass
else:
self.test_window.switch_btn['text'] = 'Pause'
return self.work()
def test_cancel(self):
# self.test_window.progressbar['value'] = 0
print(self.test_window.progressbar['value'])
self.test_window.withdraw()
self.task.cancel()
app = App()
app.mainloop()
asyncio.run(async)
.它是在Python 3.7中添加的。需要使用
get_event_loop()
和
run_until_complete()
.(@Saad指出)
关于python - Tkinter 进度条如何在模型对话框中正确实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61897484/
我正在处理一组标记为 160 个组的 173k 点。我想通过合并最接近的(到 9 或 10 个组)来减少组/集群的数量。我搜索过 sklearn 或类似的库,但没有成功。 我猜它只是通过 knn 聚类
我有一个扁平数字列表,这些数字逻辑上以 3 为一组,其中每个三元组是 (number, __ignored, flag[0 or 1]),例如: [7,56,1, 8,0,0, 2,0,0, 6,1,
我正在使用 pipenv 来管理我的包。我想编写一个 python 脚本来调用另一个使用不同虚拟环境(VE)的 python 脚本。 如何运行使用 VE1 的 python 脚本 1 并调用另一个 p
假设我有一个文件 script.py 位于 path = "foo/bar/script.py"。我正在寻找一种在 Python 中通过函数 execute_script() 从我的主要 Python
这听起来像是谜语或笑话,但实际上我还没有找到这个问题的答案。 问题到底是什么? 我想运行 2 个脚本。在第一个脚本中,我调用另一个脚本,但我希望它们继续并行,而不是在两个单独的线程中。主要是我不希望第
我有一个带有 python 2.5.5 的软件。我想发送一个命令,该命令将在 python 2.7.5 中启动一个脚本,然后继续执行该脚本。 我试过用 #!python2.7.5 和http://re
我在 python 命令行(使用 python 2.7)中,并尝试运行 Python 脚本。我的操作系统是 Windows 7。我已将我的目录设置为包含我所有脚本的文件夹,使用: os.chdir("
剧透:部分解决(见最后)。 以下是使用 Python 嵌入的代码示例: #include int main(int argc, char** argv) { Py_SetPythonHome
假设我有以下列表,对应于及时的股票价格: prices = [1, 3, 7, 10, 9, 8, 5, 3, 6, 8, 12, 9, 6, 10, 13, 8, 4, 11] 我想确定以下总体上最
所以我试图在选择某个单选按钮时更改此框架的背景。 我的框架位于一个类中,并且单选按钮的功能位于该类之外。 (这样我就可以在所有其他框架上调用它们。) 问题是每当我选择单选按钮时都会出现以下错误: co
我正在尝试将字符串与 python 中的正则表达式进行比较,如下所示, #!/usr/bin/env python3 import re str1 = "Expecting property name
考虑以下原型(prototype) Boost.Python 模块,该模块从单独的 C++ 头文件中引入类“D”。 /* file: a/b.cpp */ BOOST_PYTHON_MODULE(c)
如何编写一个程序来“识别函数调用的行号?” python 检查模块提供了定位行号的选项,但是, def di(): return inspect.currentframe().f_back.f_l
我已经使用 macports 安装了 Python 2.7,并且由于我的 $PATH 变量,这就是我输入 $ python 时得到的变量。然而,virtualenv 默认使用 Python 2.6,除
我只想问如何加快 python 上的 re.search 速度。 我有一个很长的字符串行,长度为 176861(即带有一些符号的字母数字字符),我使用此函数测试了该行以进行研究: def getExe
list1= [u'%app%%General%%Council%', u'%people%', u'%people%%Regional%%Council%%Mandate%', u'%ppp%%Ge
这个问题在这里已经有了答案: Is it Pythonic to use list comprehensions for just side effects? (7 个答案) 关闭 4 个月前。 告
我想用 Python 将两个列表组合成一个列表,方法如下: a = [1,1,1,2,2,2,3,3,3,3] b= ["Sun", "is", "bright", "June","and" ,"Ju
我正在运行带有最新 Boost 发行版 (1.55.0) 的 Mac OS X 10.8.4 (Darwin 12.4.0)。我正在按照说明 here构建包含在我的发行版中的教程 Boost-Pyth
学习 Python,我正在尝试制作一个没有任何第 3 方库的网络抓取工具,这样过程对我来说并没有简化,而且我知道我在做什么。我浏览了一些在线资源,但所有这些都让我对某些事情感到困惑。 html 看起来
我是一名优秀的程序员,十分优秀!