Только функция, которую читает длинный датчик, должна быть в отдельной ветке. Вы можете использовать queue.Queue
для получения данных из него - в приведенном ниже примере я вставил выборку значений в функцию every_one_sec
.
Функция every_one_sec
обрабатывается событиями планирования Tkinter - нет необходимости создайте для него поток.
Кроме этого, наиболее некорректной частью вашего кода был полный вызов one_wire
перед созданием потока в
t_one_wire = one_wire()
- и тот факт, что он будет работать, также требует Ткинтер перезвонит. Tkinter хочет запускать все свои события в одном потоке - так что это может привести к проблемам.
Надеюсь, приведенных ниже комментариев достаточно для лучшего понимания
import time
import threading
from queue import Queue, Empty
from tkinter import Tk
def some_ui_code():
global stop_reading
stop_reading = True
def events_every_second(): #update clock every second
right_now = time.strftime("%I:%M:%S %p")#("%H:%M:%S")
print("time is now",right_now)
try:
result = queue.get(block=False)
except Empty:
result = None
if result is not None:
# code to display the result read from the sensor in tkinter interface goes here
...
# tkinter, not a separate thread, handles this
root.after(1000, events_every_second)
def one_wire(queue):
# this is handled in a separate thread, and "does not know" about tkinter at all
while not stop_reading:
result = call_to_read_sensor() # takes 3 seconds
queue.put(result)
print("one_wire loop")
time.sleep(0.1) # actual call to time.sleep to space sensor readings, if needed.
root = Tk()
stop_reading = False
queue = Queue()
# start events_every_second - tkinter will handle the subsequent calls with the `root.after` scheduling
events_every_second()
thread_one_wire = threading.Thread(target=t_one_wire, args=(queue,))
thread_one_wire.start()
root.mainloop()