python多线程中 接收到Ctrl+C后 怎么结束正处于阻塞状态的子线程
在python多线程中,只有主线程能接收到信号,而且不能使用join阻塞,
那么主线程接收到信号后怎么结束正在阻塞状态中的子线程呢
如下代码子线程receive_task1
正阻塞在os.read()
(接收串口数据,串口没有收到数据)函数中,有没有什么办法让os.read()
函数退出呢?
如果在主线程中使用os.read()
,默认情况可以使用Ctrl+C
打断,现在这种情况如何做最佳?
import os
import signal
import threading
import time
def test(signum, frame):
print("\nreceived sig %d" % (signum))
print("how to make os.read(...) in rcv_task1 thread stop? ")
print("except use sys.exit(1)")
def receive_task1():
fd = os.open("/dev/ttyUSB1", os.O_RDWR)
os.read(fd, 10)
print("receive thread end")
signal.signal(signal.SIGINT, test)
rcv_task1 = threading.Thread(target=receive_task1)
rcv_task1.setDaemon(True)
rcv_task1.start()
while(1):
time.sleep(1)
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
首先我没有认真看你的代码(因为我打算只给你提供正确的思路)。
然后,这里是这个问题的答案:
信号与线程的关系:
收到了信号如何退出子线程?
子线程的状态只能子进程自己控制。 其它线程对它只有两个可用操作(
.join
和.is_alive
)。但是线程间是可以通信的(调度)。通过
queue.Queue
子线程就能够收到其它线程期望它的行为。比如主线程收到 Ctrl+C(SIGINT) 后,放置 'quit' 消息在 QUEUE 中,然后子线程从 QUEUE 获取到了 'quit' 消息后,就要自己做退出动作。
所以为了实现这一点,你需要非常小心地编写线程(如果你的线程有阻塞的可能性并且你想要线程一直在后台运行):
如果你需要实现一个一直在后台运行的线程,那么你就不应该使用函数的方式编写线程!所以我的建议是编写一个类级的线程。这样的话你就可以将退出机制和循环机制分离出来了。
由于上述第一点的原因,所以在编写可能会阻塞的类级线程上,没有“通用”的解决方法:socket 有设置 socket 的不阻塞方式,queue.Queue 有自己的设置不阻塞的方式。
(关于什么是阻塞IO,什么是非阻塞 IO,可以看 APUE 有一章涉及。)
最后我提供我的一篇 CSDN 博客:【Python】【多线程】Python 多线程开发指南|完成度:20%
截至 2019/02/27 该文章完成 20%,但是上述所有内容都能够在该博客中找到。
我只学过socket库,不熟悉os库。以我目前掌握的知识,解决办法有两个:
1、改成多进程模式,然后在主进程中terminate掉子进程即可;
2、将子线程中的文件描述符fd设置成非阻塞模式,正好查到os库中有os.set_blocking(fd, blocking)函数可供使用。
代码大致如下(我没在Linux下测试过):