Как правильно завершить поток Python3 во время чтения потока - PullRequest
5 голосов
/ 05 мая 2011

Я использую поток для чтения строк из потока (/ dev / tty1) при обработке других вещей в основном цикле. Я хотел бы, чтобы поток завершался вместе с основной программой при нажатии CTRL-C.

   from threading import Thread

   class myReader(Thread):
      def run(self):
         with open('/dev/tty1', encoding='ascii') as myStream:
            for myString in myStream:
               print(myString)
      def quit(self):
         pass # stop reading, close stream, terminate the thread

   myReader = Reader()
   myReader.start()
   while(True):
      try:
         pass # do lots of stuff
      KeyboardInterrupt:
         myReader.quit()
         raise

Обычное решение - логическая переменная внутри цикла run () - здесь не работает. Какой рекомендуемый способ борьбы с этим?

Я могу просто установить флаг Daemon, но тогда я не смогу использовать метод quit (), который может оказаться полезным в будущем (для некоторой очистки). Есть идеи?

Ответы [ 2 ]

4 голосов
/ 12 сентября 2011

Сделать вашу нить нить демона .После выхода из всех потоков, не являющихся демонами, программа завершается.Поэтому, когда Ctrl-C передается вашей программе и основной поток завершается, нет необходимости явно уничтожать программу чтения.

    myReader = Reader()
    myReader.daemon = True
    myReader.start()
4 голосов
/ 15 июня 2011

AFAIK, в Python 3 нет встроенного механизма для этого (как и в Python 2).Вы пробовали проверенный подход Python 2 с PyThreadState_SetAsyncExc, документированные здесь и здесь , или альтернативный подход трассировки здесь ?

Вотслегка измененная версия подхода PyThreadState_SetAsyncExc сверху:

import threading
import inspect
import ctypes 

def _async_raise(tid, exctype):
    """raises the exception, performs cleanup if needed"""
    if not inspect.isclass(exctype):
        exctype = type(exctype)
    res = ctypes.pythonapi.PyThreadState_SetAsyncExc(ctypes.c_long(tid), ctypes.py_object(exctype))
    if res == 0:
        raise ValueError("invalid thread id")
    elif res != 1:
        # """if it returns a number greater than one, you're in trouble, 
        # and you should call it again with exc=NULL to revert the effect"""
        ctypes.pythonapi.PyThreadState_SetAsyncExc(tid, None)
        raise SystemError("PyThreadState_SetAsyncExc failed")

def stop_thread(thread):
    _async_raise(thread.ident, SystemExit)
...