Python网络编程之通过ThreadingMixIn实现多线程异步套接字程序

Original 2016-11-07 11:15:07 616
abstract:使用ThreadingMixIn类的特点: 在线程间共享应用的状态,与使用ForkingMixIn类相比避免进程间通信的复杂操作。代码如下:#! /usr/bin/python    import os  import socket  import threading  import SocketS

使用ThreadingMixIn类的特点: 在线程间共享应用的状态,与使用ForkingMixIn类相比避免进程间通信的复杂操作。

代码如下:

#! /usr/bin/python
 
 import os
 import socket
 import threading
 import SocketServer
 
 SERVER_HOST = 'localhost'
 SERVER_PORT = 0
 BUF_SIZE = 1024
 
 def client(ip, port, message):
     """A client to test threading mixin server"""
     sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
     sock.connect((ip, port))
     try:
         sock.sendall(message)
         response = sock.recv(BUF_SIZE)
         print "Client received: %s" % response
     finally:
         sock.close()
 class ThreadedTCPRequestHandler(SocketServer.BaseRequestHandler):
     """An example of threaded TCP request handler """
     def handle(self):
         data = self.request.recv(1024)
         current_thread = threading.current_thread()
         response = "%s: %s" % (current_thread.name, data)
         self.request.sendall(response)
 
 class ThreadedTCPServer(SocketServer.ThreadingMixIn, SocketServer.TCPServer):
     """Nothing add here, inherited everything from parenets"""
     pass
 
 if __name__ == '__main__':
     #run server
     server = ThreadedTCPServer((SERVER_HOST, SERVER_PORT), ThreadedTCPRequestHandler)
     ip, port = server.server_address
     
     # start a thread with the server, one thread per request
     server_thread = threading.Thread(target=server.serve_forever)
     # Exit the server thread when the main thread exits
     server_thread.daemon = True
     server_thread.start()
     print "Server loop running on thread: %s" % server_thread.name
 
     # run clients
     client(ip, port, "Hello from client1")
     client(ip, port, "Hello from client2")
     client(ip, port, "Hello from client3")
 
     #server cleanup
     server.shutdown()


注意1:在写ThreadedTCPRequestHandler类中的方法handle的时候将handle写为了handler,结果在执行程序后,显示服务器返回给客户端的信息为空,原因是handle方法是ThreadedTCPRequestHandler类所继承的BaseRequestHandler类的类方法,由于BaseRequestHandler类的handle方法默认行为是啥也不做,所以要覆写handle方法来处理接受客户端消息,处理客户端请求,如果写为handler则在程序中默认调用了父类BaseRequestHandler的handle方法,但是啥也没做,所以显示服务器返回信息为空。

RequestHandler.handle()

This function must do all the work required to service a request. The default implementation does nothing. Several instance attributes are available to it; the request is available as self.request; the client address as self.client_address; and the server instance as self.server, in case it needs access to per-server information.

The type of self.request is different for datagram or stream services. For stream services, self.request is a socket object; for datagram services, self.request is a pair of string and socket. However, this can be hidden by using the request handler subclasses StreamRequestHandler or DatagramRequestHandler, which override the setup() and finish() methods, and provide self.rfile and self.wfile attributes. self.rfile and self.wfile can be read or written, respectively, to get the request data or return data to the client.

注意2,关于线程对象的setDaemon属性,为True表示服务器线程为后台线程,当没有活动连接到后台线程时,程序执行完毕自动退出,当没有活动的非后台线程活动时,执行完毕程序才会自动退出。对于本代码,具体表现是如果在setDaemon为True的状态,不执行服务器套接字的shutdown(),程序执行完一样会自动退出,而setDaemon为False的状态, 则一定要执行shutdown(),程序才会自动终止。


Release Notes

Popular Entries