python threading中停止socket.recv

在使用PyQt5进行socket局域网通信时遇到窗口关闭后程序仍然运行的问题,原因是socket.close()后socket.recv()仍堵塞线程。通过设置socket超时解决了这一问题,当超过设定时间未接收到数据时,会触发socket.timeout异常,从而避免线程无法释放。修改后的代码在socket.recv()中增加了超时处理,确保了线程的正常退出。
摘要由CSDN通过智能技术生成

为无法停止threading的坑做一次记录

在用PyQt5界面库写socket局域网通信时,发现窗口关闭后程序仍在运行,认真检查后是线程中的socket的recv函数出了问题,在socket关闭后(socket.close()),socket.recv()仍会堵塞,导致线程无法释放,现找到了三种做法

修改前服务器部分代码如下

threading.Thread(target=self.connect).start()

def connect(self, server=None):
    # 等待用户连接
    if not server:
        server = self.server  # socket
    # 服务器要一直运行所以使用死循环
    while True:
        # 接受所连接的客户端的信息
        try:
            connect, addrss = server.accept()
        except OSError:
            break  # 服务器关闭
        # 每连接一个客户端就开启一个线程
        t = threading.Thread(target=self.member_receive, args=(connect, addrss))
        t.start()
        self.receive_thread.append(t)

def member_receive(self, connect, addrss):
    while True:
        try:
            rData = connect.recv(1024)
        except Exception as error:
            print(error)
            break

 然后member_receive中的connect.recv(1024)会一直堵塞运行

修改方法来自 如何从Python中的另一个线程中止socket.recv()

使用超时的解决方案。这将中断recv(实际上在超时之前),这很好:

链接中的代码:

# Echo server program
import socket
from threading import Thread
import time


class ClientThread(Thread):
    def __init__(self, clientSocke):
        Thread.__init__(self)
        self.clientSocket = clientSocket

    def run(self):
        while 1:
            try:
                data = self.clientSocket.recv(1024)
                print "Got data: ", data
                self.clientSocket.send(data)
            except socket.timeout: 
                # If it was a timeout, we want to continue with recv
                continue
            except:
                break

        self.clientSocket.close()

HOST = ''
PORT = 6000
serverSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
serverSocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
serverSocket.bind((HOST, PORT))
serverSocket.listen(1)

clientSocket, addr = serverSocket.accept()
clientSocket.settimeout(5)

print 'Got a new connection from: ', addr
clientThread = ClientThread(clientSocket)
clientThread.start()

# Close it down immediatly 
clientSocket.close()

先设置超时的时间为5秒

socket.settimeout(5)

即1秒内如果没有接收到就会报出socket.timeout,然后跳到下一次循环,从而避免程序堵塞、线程无法释放

修改后服务器代码如下

self.server.settimeout(5)
threading.Thread(target=self.connect).start()

def connect(self, server=None):
    # 等待用户连接
    if not server:
        server = self.server  # socket
    # 服务器要一直运行所以使用死循环
    while True:
        # 接受所连接的客户端的信息
        try:
            connect, addrss = server.accept()
        except OSError:
            break  # 服务器关闭
        # 每连接一个客户端就开启一个线程
        t = threading.Thread(target=self.member_receive, args=(connect, addrss))
        t.start()
        self.receive_thread.append(t)

def member_receive(self, connect, addrss):
    while True:
        try:
            rData = connect.recv(1024)
        except socket.timeout:
            break
        except Exception as error:
            print(error)
            break

当然,你得确保你的超时时间足够使recv函数接收完所有的数据才行。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值