Python版单进程、多进程、多线程服务器

来源:互联网 发布:天谕浅粉色头发数据 编辑:程序博客网 时间:2024/06/14 00:24

在学习了Python的网络编程以及计算机网络的相关知识之后,编写了Python版的服务器。有三种实现方式,分别是单进程、多进程、多线程。
实现原理:建立tcp套接字,为该套接字绑定本机信息,之后进入循环,令tcp套接字接收信息,并传递给客户端套接字以及收集客户端地址,然后再次进入循环,服务器套接字不断接收客户端传递进来的信息,判断信息(字符串)长度是否为零,若为0则表示客户端已经关闭。
单进程的较为简单,且不太现实,没有实现意义,但是有助于理解服务器的工作方式。
单进程版如下:

#-*- coding:utf-8 -*-from socket import *#单进程服务器serSocket = socket(AF_INET, SOCK_STREAM)serSocket.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)#表示重复利用套接字信息localAddr = ("", 7789)serSocket.bind(localAddr)serSocket.listen(5)while True:    print("----主进程,等待客户端连接")    newSocket, destAdree = serSocket.accept()    print("----主进程,接下来负责数据处理[%s]"%(str(destAdree)))    try:        while True:            revDate = newSocket.recv(1024)            if len(revDate) > 0:                print("数据是:[%s]"%str(revDate.decode("utf-8")))            else:                print("it's nothing")                break    finally:        newSocket.close()serSocket.close()

多进程版如下:

from socket import *from multiprocessing import *from time import sleep# 处理客户端的请求并为其服务def dealWithClient(newSocket,destAddr):    while True:        recvData = newSocket.recv(1024)        if len(recvData)>0:            print('recv[%s]:%s'%(str(destAddr), recvData))        else:            print('[%s]客户端已经关闭'%str(destAddr))            break    newSocket.close()def main():    serSocket = socket(AF_INET, SOCK_STREAM)    serSocket.setsockopt(SOL_SOCKET, SO_REUSEADDR  , 1)    localAddr = ('', 7788)    serSocket.bind(localAddr)    serSocket.listen(5)    try:        while True:            print('-----主进程,,等待新客户端的到来------')            newSocket,destAddr = serSocket.accept()            print('-----主进程,,接下来创建一个新的进程负责数据处理[%s]-----'%str(destAddr))            client = Process(target=dealWithClient, args=(newSocket,destAddr))            client.start()            #因为已经向子进程中copy了一份(引用),并且父进程中这个套接字也没有用处了            #所以关闭            newSocket.close()    finally:        #当为所有的客户端服务完之后再进行关闭,表示不再接收新的客户端的链接        serSocket.close()if __name__ == '__main__':    main()

在编写中要注意循环的写法、位置,以及异常的处理。
多线程版如下:

#-*- coding:utf-8 -*-from socket import *from threading import Thread from time import sleep#多线程服务器def dealwithClient(newSocket, destAddr):    while True:        newMassage = newSocket.recv(1024)        if len(newMassage) > 0: #此处注意不能写成>=            print("new massage from [%s] is %s"%(destAddr, newMassage.decode("utf-8")))        else:            print("client was closed")            break    newSocket.close()def main():    serSocket = socket(AF_INET, SOCK_STREAM)    serSocket.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)    localAddr = ('', 7788)    serSocket.bind(localAddr)    serSocket.listen(5)    try:        while True:            print("---wait for client---")            newSocket, destAddr = serSocket.accept()            print("---I have accepted a client---")            client = Thread(target = dealwithClient, args = (newSocket, destAddr))            client.start()            #因为线程中共享这个套接字,如果关闭了会导致这个套接字不可用,            #但是此时在线程中这个套接字可能还在收数据,因此不能关闭            #newSocket.close()     finally:        serSocket.close()if __name__ == "__main__":    main()

线程的开销较小,且共享套接字。

原创粉丝点击