Python select实现socket并发

Python select 
  Python的select()方法直接调用操作系统的IO接口,它监控sockets,open files, and pipes(所有带fileno()方法的文件句柄)何时变成readable 和writeable, 或者通信错误,select()使得同时监控多个连接变的简单,并且这比写一个长循环来等待和监控多客户端连接要高效,因为select直接通过操作系统提供的C的网络接口进行操作,而不是通过Python的解释器。
注:使用Python的文件对象选择()为Unix工作,但在Windows下不支持。
 
select实现socketserver
注:接下来通过echo server例子要以了解select 是如何通过单进程实现同时处理多个非阻塞的socket连接的
注:必须在非阻塞情况下。
 
服务端:
import select
import socket
import queue

server = socket.socket()
server.bind(('localhost' ,9000))
server.listen(1000)

# 设置非阻塞模式
server.setblocking(False)
msg_dic = {}

# 内核检测并发链接
inputs = [server,]

# outputs存放链接需要返回的数据
outputs = []

# 为了循环调入select加入while
while True:

    # 1,需要内核检测哪些链接,有一个活动就返回所有链接循环。
    # 2,处理返回数据。
    # 3,如果有并发的链接断开,内核会返回报错到inputs内,有哪几个有问题。
    # 有链接进入会返回三个数据:
    # readable:返回一个列表,活动的,可读数据的
    # writeable:存放需要返回的数据。
    # exceptional:返回出现异常的活动链接
    readable,writeables,exceptional = select.select(inputs,outputs,inputs)
    print(readable,writeables,exceptional)

    # 收处理
    for r in readable:
        # 代表来了一个新连接
        if r is server:

            # 等待客户端生成实例
            conn,addr = server.accept()

            # 打印客户端IP地址
            print("来了个新连接",addr)

            #   因为这个新建里的链接还没发数据过来,现在就收数据的话程序就报错了
            #所以要想实现客户端发数据来时server端能知道,就需要让select在检测这
            #个conn。
            inputs.append(conn)

            # 初始化一个队列,后面存要返回这个客户端的数据
            msg_dic[conn] = queue.Queue()

        # 接收新连接数据
        else:
            # 获取数据
            data = r.recv(1024)
            # 打印数据
            print("收到数据:",data)

            # 将返回的数据排列到队列中
            msg_dic[r].put(data)

            # 放入返回的链接队列
            outputs.append(r)

    # 发数据:要返回给客户端的链接列表
    for w in writeables:
# 重链接列表中取出队列的实例 data_to_client = msg_dic[w].get()
# 返回给客户端数据 w.send(data_to_client)
# 确保下次循环的时候writeable,不返回这个已经处理完的链接 outputs.remove(w) # 删除:错误链接 for e in exceptional:
# 查找错误链接是否存在outputs if e in outputs:
# 如果有就删除错误链接 outputs.remove(e)
# 删除inputs下的错误链接 inputs.remove(e)
# 删除队列中的错误链接 del msg_dic[e]

 

客户端:

import socket
client = socket.socket()
#client.connect(('192.168.1.177',9999))
client.connect(('localhost',9000))
while True:
    msg = input(">>:").strip()
    if len(msg) == 0:continue
    client.send(msg.encode("utf-8"))
    data = client.recv(1024)
    print("recv:",data.decode())
client.close()

 

select实现socketserver并发(完整版)
import select
import socket
import sys
import queue


server = socket.socket()
server.setblocking(0)

server_addr = ('localhost',10000)

print('starting up on %s port %s' % server_addr)
server.bind(server_addr)

server.listen(5)


inputs = [server, ] #自己也要监测呀,因为server本身也是个fd
outputs = []

message_queues = {}

while True:
    print("waiting for next event...")

    readable, writeable, exeptional = select.select(inputs,outputs,inputs) #如果没有任何fd就绪,那程序就会一直阻塞在这里

    for s in readable: #每个s就是一个socket

        if s is server: #别忘记,上面我们server自己也当做一个fd放在了inputs列表里,传给了select,如果这个s是server,代表server这个fd就绪了,
            #就是有活动了, 什么情况下它才有活动? 当然 是有新连接进来的时候 呀
            #新连接进来了,接受这个连接
            conn, client_addr = s.accept()
            print("new connection from",client_addr)
            conn.setblocking(0)
            inputs.append(conn) #为了不阻塞整个程序,我们不会立刻在这里开始接收客户端发来的数据, 把它放到inputs里, 下一次loop时,这个新连接
            #就会被交给select去监听,如果这个连接的客户端发来了数据 ,那这个连接的fd在server端就会变成就续的,select就会把这个连接返回,返回到
            #readable 列表里,然后你就可以loop readable列表,取出这个连接,开始接收数据了, 下面就是这么干 的

            message_queues[conn] = queue.Queue() #接收到客户端的数据后,不立刻返回 ,暂存在队列里,以后发送

        else: #s不是server的话,那就只能是一个 与客户端建立的连接的fd了
            #客户端的数据过来了,在这接收
            data = s.recv(1024)
            if data:
                print("收到来自[%s]的数据:" % s.getpeername()[0], data)
                message_queues[s].put(data) #收到的数据先放到queue里,一会返回给客户端
                if s not  in outputs:
                    outputs.append(s) #为了不影响处理与其它客户端的连接 , 这里不立刻返回数据给客户端


            else:#如果收不到data代表什么呢? 代表客户端断开了呀
                print("客户端断开了",s)

                if s in outputs:
                    outputs.remove(s) #清理已断开的连接

                inputs.remove(s) #清理已断开的连接

                del message_queues[s] ##清理已断开的连接


    for s in writeable:
        try :
            next_msg = message_queues[s].get_nowait()

        except queue.Empty:
            print("client [%s]" %s.getpeername()[0], "queue is empty..")
            outputs.remove(s)

        else:
            print("sending msg to [%s]"%s.getpeername()[0], next_msg)
            s.send(next_msg.upper())


    for s in exeptional:
        print("handling exception for ",s.getpeername())
        inputs.remove(s)
        if s in outputs:
            outputs.remove(s)
        s.close()

        del message_queues[s]
posted @ 2018-01-06 19:49  kevin.Xiang  阅读(2688)  评论(0编辑  收藏  举报