如何使用队列在python3中的tkinter中进行线程化?

时间:2019-02-01 03:52:59

标签: python-3.x multithreading tkinter queue

我正在尝试使用专门在python3中的队列在tkinter中定义线程

我在python2中有类似的代码,使用类似的方法可以很好地工作而无需排队,但是从我读过的python3中,tkinter不允许使用gui多线程。我发现了一些使用队列过程的示例。他们概述了我想创建一个Queue对象,一个可以访问该队列的新线程,并检查主线程中的输入

#!/usr/bin/python3

from tkinter import *
import time
import threading
import queue
import subprocess

def temp_sensor(queue_list):
    warning = 0
    while True:
        var = "cat /sys/class/thermal/thermal_zone*/temp"
        temp_control = subprocess.check_output([var], shell=True)
        temp_length = len(temp_control)
        temp_control = temp_control[35:]
        temp_control = temp_control[:-4]
        temp_control = int(temp_control)
        degree_sign= u'\N{DEGREE SIGN}'
        displayed_temp = "Tempature: " + str(temp_control) + degree_sign + "C"

        if temp_control > 79:
            warning = warning + 1
            if warning == 3:
                print ("Warning Core Tempature HOT!")
                warning = 0

        if temp_control > 90:
                time.sleep(3)
                print ("Warning EXTREMLY to HOT!!!")

        queue_list.put(displayed_temp)            
        time.sleep(1)

class Gui(object):
    def __init__(self, queue_list):
        self.queue_list = queue_list
        self.root = Tk()
        self.root.geometry("485x100+750+475")
        main_tempature_status = StringVar(self.root)

        Ts = Entry(self.root, textvariable=main_tempature_status)
        Ts.pack()
        Ts.place(x=331, y=70, width=160, height=25)
        Ts.config(state=DISABLED, disabledforeground="Black")

        self.root.after(1000, self.read_queue)

    def read_queue(self):
        try:
            temp = self.queue.get_nowait()
            self.main_tempature_status.set(temp)
        except queue_list.Empty:
            pass

        self.root.after(1000, self.read_queue)

if __name__ == "__main__":
    queue_list = queue.Queue()

    gui = Gui(queue_list)
    t1 = threading.Thread(target=temp_sensor, args=(queue_list,))
    t1.start()

    gui.root.mainloop()

我想要的结果是运行这些定义中的一些来执行各种任务,并使用python3在tkinter条目中显示其变量。 当我运行我的代码时,它为我提供了队列中的变量,但它不会发布到GUI。请原谅我那么少的pythonic代码。

1 个答案:

答案 0 :(得分:0)

将您的<buttons:SfRadioGroup Grid.Row="3" Grid.Column="0" x:Name="radioGroup" HorizontalOptions="Fill" Orientation="Vertical" VerticalOptions="CenterAndExpand" Margin="10,5,10,0"> <buttons:SfRadioButton x:Name="Sum" TextColor="Gray" IsChecked="{Binding Option2, Mode=TwoWay}" Text="Sum" VerticalOptions="Center" Margin="1,1,1,1" /> <Label Text="detail Sum" BackgroundColor="Gray" BindingContext="{x:Reference Sum}" IsVisible="{Binding IsChecked}"/> <buttons:SfRadioButton x:Name="Individual" TextColor="Blue" IsChecked="{Binding Option1, Mode=TwoWay}" Text="Individual" VerticalOptions="Center" Margin="1,1,1,1" StateChanged="SfRadioButton_StateChanged"/> <Label Text="detail Individual" BackgroundColor="Gray" BindingContext="{x:Reference Individual}" IsVisible="{Binding IsChecked}"/> </buttons:SfRadioGroup> 类更改为此:

Gui

说明

  1. 变量class Gui(object): def __init__(self, queue_list): self.queue_list = queue_list self.root = Tk() self.root.geometry("485x100+750+475") self.main_tempature_status = StringVar(self.root) self.Ts = Entry(self.root, textvariable=self.main_tempature_status) self.Ts.pack() self.Ts.place(x=331, y=70, width=160, height=25) self.Ts.config(state=DISABLED, disabledforeground="Black") self.root.after(1000, self.read_queue) def read_queue(self): try: temp = self.queue_list.get_nowait() self.Ts.config(state=NORMAL) self.main_tempature_status.set(temp) self.Ts.config(state=DISABLED) except queue.Empty: pass self.root.after(1000, self.read_queue) 在函数main_temperature_status中用作类变量,但未定义为类变量。
  2. 如果read_queue小部件的值更改始终处于禁用状态,则无法显示它,因此请在Entry中更改值之前启用它。