Потоковый класс Python вызывает другой поточный класс (помощь в очереди)

Я пытаюсь управлять 3-осевым принтером с помощью контроллера x-box. Чтобы получить входные данные из x-box, я позаимствовал код из martinohanlon https://github.com/martinohanlon/XboxController/blob/master/XboxController.py Я также создал код, который читает текстовый файл построчно (G-код) переместить принтер.

Я хотел бы иметь возможность использовать контроллер X-Box, чтобы выбрать файл G-кода и запустить его, а затем, пока принтер работает, продолжайте прослушивать кнопку отмены только в случае, если печать идет неправильно. Контроллер - это многопоточный класс, а мой readGcode - это многопоточный класс.

У меня проблема в том, что когда я использую контроллер для запуска класса readGcode, я не могу связаться с контроллером, пока этот поток не закончится.

Мое временное решение - использовать контроллер для выбора файла, а затем передать путь этих файлов в класс readGcode. В классе readGcode он продолжает пытаться открыть файл, используя блок try, и терпит неудачу, пока путь к файлу не будет приемлемым. Затем он меняет bool, что заставляет его пропустить дальнейшее чтение до его завершения.

Код:

import V2_Controller as Controller
import V2_ReadFile as Read
import time
import sys
# file daialogue
import tkinter as tk
from tkinter import filedialog

# when X is selected on the x-box controller 
def X(xValue):
    if not bool(xValue):
        try:
            f=selectfile()
            rf.setfilename(f)
        except:
            print("failed to select file")

def selectfile():
    try:
        root = tk.Tk()  # opens tkinter
        root.withdraw()  # closes the tkinter window
        return filedialog.askopenfilename()
    except Exception:
        print("no file")

# setup xbox controller
xboxCont = Controller.XboxController(controlCallBack, deadzone=30, 
scale=100, invertYAxis=True)
# init the readfile class
rf = Read.Readfile()

# set the custom function for pressing X
xboxCont.setupControlCallback(xboxCont.XboxControls.X, X)

try:
    # start the controller and readfile threads
    xboxCont.start()
    rf.start()

    xboxCont.join()
    rf.join()

    while True:
        time.sleep(1)

# Ctrl C
except KeyboardInterrupt:
    print("User cancelled")

# error
except:
    print("Unexpected error:", sys.exc_info()[0])
    raise

finally:
    # stop the controller
    xboxCont.stop()
    rf.stop()

V2_Readfile

# Main class for reading the script
class Readfile(threading.Thread):

    # supports all variables needed to read a script
    class readfile:
        fileselect = True
        linecount = 0
        currentline = 0
        commands = []

    # setup readfile class
    def __init__(self):
        # setup threading
        threading.Thread.__init__(self)
        # persist values
        self.running = False
        self.reading = False

    def setfilename(self,filename):
        self.filename = filename

    # called by the thread
    def run(self):
        self._start()

    # start reading
    def _start(self):
        self.running = True
        while self.running:
            time.sleep(1)
            if not self.reading:
                try:
                    self.startread()
                except:
                    pass

    def startread(self):
        try:
            with open(self.filename, "r") as f:  # read a local file
                f1 = f.readlines()
                # run through each line and extract the command from each line
                linecount = 0
                line = []
                for x in f1:
                    # read each line into an array
                    line.append(x.split(";")[0])
                    linecount += 1

                # Store the variables for later use
                self.readfile.linecount = linecount
                self.readfile.commands = line
                self.reading = True
        except Exception:
            pass

        i = 0
        while i < self.readfile.linecount and self.reading:
            self.readfile.currentline = i + 1
            self.readline(i)
            i += 1

        # the following stops the code from reading again
        self.reading = False
        self.filename = ""


    def readline(self,line):
        Sort.sortline(self.readfile.commands[line])

    # stops the controller
    def stop(self):
        self.running = False

1 ответ

Решение

Вы можете использовать примитив синхронизации, такой как threading.Event.

Для этого вам нужно изменить Readfile класс как это:

from threading import Event


class Readfile(threading.Thread):

    # setup readfile class
    def __init__(self):
        # setup threading
        threading.Thread.__init__(self)
        # persist values
        self.running = False
        self.reading = False
        self.reading_file = Event()  # Initialize your event here it here

    def setfilename(self,filename):
        self.filename = filename
        self.reading_file.set()  # This awakens the reader

    def _start(self):
        self.running = True
        self.reading_file.wait()  # Thread will be stopped until readfilename is called
        self.startread()

Другой примитив синхронизации, который стоит изучить queue.Queue, Это может быть полезно, если вы хотите обработать более одного имени файла.

Шаблон, который вы описываете в своем вопросе, называется " Ожидание при занятости", и его следует по возможности избегать.

Другие вопросы по тегам