Linux编程之select

写了python脚本测试,发现正常的selectors、socketserver方式的服务器,在WINDOS下最多同时连接512个客户端,linux服务器下据说最多同时连接1024个客户端。想连接更多客户端只能采用gevet协程模式。

原理是操作系统底层就支持这么多。

以select写的聊天服务器为例:

select服务器:

select_server.py代码:

# coding=utf-8

import select
import socket
import threading
from logic.c_msg import CMsg
from logic.c_send import CSend


class CSelect(threading.Thread):

    def __init__(self, lock, users):
        threading.Thread.__init__(self)
        self.mLock = lock
        self.mUsers = users

    def run(self):
        server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        server.setblocking(False)
        host = socket.gethostname()
        print("host = {}".format(host))
        server.bind(("0.0.0.0", 9090))
        server.listen(5)  #最多五人等待

        inputs = [server]
        outputs = []
        uid = 0

        while True:
            rlist, wlist, elist = select.select(inputs, outputs, inputs)
            #NOTE: 没有客户端连接时,一直锁定在此。一旦有客户端连接上了,就一直循环检测,不停打印下面的语句
            # print("len(rlist)={}, len(wlist)={}, len(elist)={}".format(len(rlist), len(wlist), len(elist)))
            # 遍历可读
            for r in rlist:
                if r is server:
                    conn, addr = r.accept()
                    conn.setblocking(False)
                    inputs.append(conn)
                    outputs.append(conn)
                    uid += 1
                    self.mUsers[conn] = CMsg(conn, addr, uid)
                    print("accept: {},{}".format(uid, conn))
                else:

                    try:
                        data = r.recv(1024)
                        if (data):
                            self.mLock.acquire()
                            txt = data.decode("utf-8")
                            print("recv: txt={}, r={} ".format(txt, r))
                            user = self.mUsers[r]
                            content = "[用户({})]:{}".format(user.mUid, txt)
                            user.addContent(content)
                            self.mLock.release()

                        else:  # 数据为空,则是客户端断开
                            print("断开 {}".format(r))
                            if r in outputs:
                                outputs.remove(r)
                            inputs.remove(r)
                            r.close()
                            self.mLock.acquire()
                            self.mUsers[r].reset()
                            del self.mUsers[r]
                            self.mLock.release()

                    except ConnectionResetError as e:
                        print("用户断开:r={},  e={}".format(r, e))
                        if r in outputs:
                            outputs.remove(r)
                        inputs.remove(r)
                        r.close()

                        self.mLock.acquire()
                        self.mUsers[r].reset()
                        del self.mUsers[r]
                        self.mLock.release()
                    finally:
                        pass

            self.mLock.acquire()
            # 遍历可写
            for conn, msg in self.mUsers.items():
                msg.mIsConnected = False

            for w in wlist:
                if w in self.mUsers:
                    self.mUsers[w].mIsConnected = True

            for s in elist:
                if (s in self.mUsers):
                    self.mUsers[s].reset()
                    del self.mUsers[s]

                (s in inputs) and inputs.remove(s)
                (s in outputs) and outputs.remove(s)
                s.close()
            self.mLock.release()


if __name__ == "__main__":
    lock = threading.Lock()
    users = {}
    sel = CSelect(lock, users)
    sel.start()

    send = CSend(lock, users)
    send.start()

    while True:
        line = input("输入:")
        if line == 'q':
            break

c_msg.py代码:

# coding=utf-8

from queue import Queue


class CMsg():

    def __init__(self, so, addr, uid=0):
        self.mSock = so
        self.mAddr = addr
        self.mContentQu = Queue()
        self.mUid = uid
        self.mIsConnected = True

    def addContent(self, content):
        self.mContentQu.put(content)

    def reset(self):
        self.mContentQu.queue.clear()
        self.mUid = 0
        self.mIsConnected = True

c_send.py代码:<

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux网络编程中的select函数用于在一组文件描述符上进行多路复用,以便能够同时监视多个文件描述符是否有数据可读、可写或异常等事件发生。 select函数的原型如下: ```c #include <sys/select.h> int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout); ``` 其中,参数说明如下: - nfds:需要监视的文件描述符集合中最大描述符加1的值。 - readfds:可读事件的文件描述符集合。 - writefds:可写事件的文件描述符集合。 - exceptfds:异常事件的文件描述符集合。 - timeout:超时时间,如果设置为NULL则表示一直等待,直到有事件发生。 select函数返回值表示有事件发生的文件描述符个数,若返回值为-1则表示发生错误,可通过errno来获取具体错误信息。 使用select函数的基本步骤如下: 1. 创建并初始化文件描述符集合readfds、writefds和exceptfds,并将需要监视的文件描述符添加到相应的集合中。 2. 调用select函数,等待事件发生。 3. 检查select函数的返回值,判断有事件发生的文件描述符个数。 4. 遍历文件描述符集合,检查每个文件描述符对应的就绪事件类型。 5. 处理事件。 select函数在网络编程中常用于实现多客户端并发处理、IO复用等功能。它可以同时监听多个文件描述符,避免了使用多线程或多进程的复杂性和性能开销。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值