Python之异步IO模型实现socket并发

服务端:

import socket
sk = socket.socket()
addr = ('127.0.0.1',8090)
sk.bind(addr)
sk.listen()
sk.setblocking(False) # 设置非阻塞模式,accept,recv等方法均不阻塞,而是抛出异常

conn_lst = [] # 建立一个列表存储接收到的连接
del_lst = [] # 建立一个列表存储已经关闭的连接
while True:
    try:
        conn,addr_client = sk.accept() # 没有连接时--BlockingIOError
        print('接收到一个连接') # 如果上一步没有异常发生,则表示收到一个连接
        conn_lst.append(conn) # 将接收到的连接放入列表中
    except BlockingIOError:
        for conn in conn_lst: # 循环接收到的连接,尝试接收消息
            try:
                msg = conn.recv(1024) # 如果没有收到,同样抛出异常--BlockingIOError
                if not msg: # 从一个关闭的客户端接收消息,收到的值为空
                    del_lst.append(conn) # 将已关闭的连接放入待删除列表中
                    continue
                print(msg)
            except BlockingIOError:
                for conn in del_lst: # 将客户端已关闭的连接关闭回收
                    conn.close()
                    conn_lst.remove(conn)
                del_lst.clear() # 清空已关闭的连接记录

客户端:

import socket
from threading import Thread
import time
def client():
    sk = socket.socket()
    addr = ('127.0.0.1',8090)
    sk.connect(addr)
    sk.send(b'hell0')
    sk.close()
if __name__ == '__main__': # 模拟20个客户端并发连接服务端
    for i in range(20):
        Thread(target=client).start()

弊端:while True循环模式极耗CPU,通常需要在循环中加入sleep,但是加入sleep影响实时性

IO多路复用模型-服务端

# 鉴于异步IO模型while True不停扫描高消耗CPU,所以采用IO多路复用模型
import select
import socket

sk = socket.socket()
addr = ('127.0.0.1',8090)
sk.setblocking(False)
sk.bind(addr)
sk.listen()

read_lst = [sk]
while True:
    # select 监听请求对象,如果没有收到请求,则阻塞,此处相当于监听accept事件
    rret,wret,xret = select.select(read_lst,[],[]) # rret为响应事件对象,若响应accept事件,则返回sk,若响应recv,则返回conn
    # print(rret)
    for i in rret:
        if i is sk:
            conn,addr = i.accept()
            read_lst.append(conn)
        else:
            msg = i.recv(1024)
            if not msg:
                read_lst.remove(i)
                i.close()
                continue
            print(msg)

 

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Socket 异步 I/O 是一种处理网络通信的方法,它允许同时处理多个连接,而不会阻塞主线程。通过使用异步 I/O,可以实现高效的并发网络编程。 在 Python 中,可以使用内置的 `socket` 模块结合 `asyncio` 库来实现 Socket 异步 I/O。下面是一个简单的示例代码: ```pythonimport asyncioimport socketasync def handle_client(client_socket): while True: data = await loop.sock_recv(client_socket,1024) if not data: break # 处理接收到的数据 response = data.upper() await loop.sock_sendall(client_socket, response) client_socket.close() async def run_server(): server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR,1) server_socket.bind(('localhost',8888)) server_socket.listen(5) while True: client_socket, _ = await loop.sock_accept(server_socket) loop.create_task(handle_client(client_socket)) loop = asyncio.get_event_loop() loop.run_until_complete(run_server()) ``` 在上面的示例中,我们创建了一个异步函数 `handle_client`,用于处理客户端的连接。`handle_client` 函数使用 `loop.sock_recv` 和 `loop.sock_sendall` 方法来实现异步的接收和发送数据操作。 然后,我们创建了一个异步函数 `run_server`,用于启动服务器并监听客户端连接。在 `run_server` 函数中,我们使用 `loop.sock_accept` 方法来等待客户端的连接,并使用 `loop.create_task` 方法创建一个任务来处理每个客户端连接。 最后,我们通过调用 `loop.run_until_complete(run_server())` 来运行服务器。 这只是一个简单的示例,实际使用中可能需要更多的逻辑来处理连接、处理数据等。但是通过使用异步 I/O,可以实现高效的网络通信和并发处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值