python11-IO多路复用

IO多路复用

  • socket在客户端与服务端建立连接后,之后的请求都需要等待
    • 原生的socket服务端只能在同一时刻处理一个请求
  • IO多路复用:
    • 可以监听多个文件描述符(socket对象),一旦文件描述符的状态出现变化,就会感知到
    • 一旦有人给服务器发送请求,服务端的socket就会发生变化
    • 或服务端通过Socket给客户端发送数据,服务端的socket也会发生变化
让socket监听多个端口
  • 原生的socket只能监听一个端口
  • 通过select模块实现socket监听多个端口
  • [注]:socket中send和sendall:send发送的内容不一定全部发送出去,返回值为发送了多少,sendall底层调用send,通过循环把所有的数据发送出去
  • IO多路复用可以接收多个文件描述符,一旦有哪个文件描述符的状态发生变化,就会帮忙处理
import socket
import select

server1 = socket.socket()
ip_port1 = ('127.0.0.1', 8001)
server1.bind(ip_port1)
server1.listen(5)

server2 = socket.socket()
ip_port2 = ('127.0.0.1', 8002)
server2.bind(ip_port2)
server2.listen(5)

print('服务器启动.........')
inputs = [server1, server2]

while True:
    # r_list:为第一个参数传入的列表,一旦这个列表中的文件描述符对象发生改变,就会把这个对象放入r_list
    # w_list:为第二个参数的列表对象关联,第二个参数的列表里有什么,w_list就有什么
    # e_list:为第三个参数的列表关联,一旦这个列表的文件描述符对象发生异常,就会把异常对象传入e_list中,通常把异常的文件描述符从监听中移除
    # 参数四:表示每隔多久扫描一次,这里为1秒
    r_list, w_list, e_list = select.select(inputs, [], [], 1)
    print('r_list大小:%d'%len(r_list))
    for sk in r_list:
        conn, addr = sk.accept()
        print('服务器接收到请求,客户端ip: %s  .port: %s' % (addr[0], addr[1]))
        conn.sendall(bytes('Hello_World!', encoding='utf-8'))
  • IO多路复用跟系统底层有关,由系统底层实现的,跟python无关,python只是通过select模块调用,window系统只支持select
  • IO多路复用发展
    • 计算机一开始只有select,大家都用select
    • select:性能比较低,底层通过for循环逐个遍历,最多支持1024个
    • poli:对select优化,个数没有限制了,但是依然不是并发的(for循环)
    • epoli:内部不再是for循环,通过异步的方式,哪个文件描述符发生变化,主动告诉系统
socket实现可以接收多个客户端访问
import socket
import select

sk1 = socket.socket()
ip_port1 = ('127.0.0.1', 8001)
sk1.bind(ip_port1)
sk1.listen(5)
print('服务端启动...........')

inputs = [sk1, ]
# 一旦有客户端访问服务端,服务端通过accept()获取到客户端的socket(conn)放入inputs列表中进行监听

while True:
    r_list, w_list, e_list = select.select(inputs, [], [], 1)
    for sk in r_list:
        if sk == sk1:
            # 一旦有人访问,sk1就会发生变化
            conn, addr = sk.accept()
            conn.sendall(bytes('Hello_World', encoding='utf-8'))
            # 把客户端的socket(conn)方法inputs监听,一旦客户端发送数据过来,conn会发生变化
            inputs.append(conn)
        else:
            # conn发生变化
            try:
                recv_bytes = sk.recv(1024)
            except Exception as e:
                inputs.remove(sk)
                print(e)
            else:
                recv_str = str(recv_bytes, encoding='utf-8')
                print(recv_str)
                sk.sendall(bytes("回复:" + recv_str, encoding='utf-8'))
            finally:
                pass

  • 在python2.7中,如果客户端断开了连接,默认会发送一个空值给服务端,服务端通过:if rev_bytes来判断是否断开连接,移除监听
  • python3.x中,客户端断开连接时变成抛异常,通过try来处理断开的逻辑
  • 上面的操作没有实现并发,比socket的优势是可以处理多个请求,但不是并发进行,通过for循环

select实现读写分离

import socket
import select

sk1 = socket.socket()
ip_port1 = ('127.0.0.1', 8001)
sk1.bind(ip_port1)
sk1.listen(5)
print('服务端启动...........')

inputs = [sk1, ]
outputs = []
message_dict = {}
# 一旦有客户端访问服务端,服务端通过accept()获取到客户端的socket(conn)放入inputs列表中进行监听

while True:
    r_list, w_list, e_list = select.select(inputs, outputs, [], 1)
    for sk in r_list:
        if sk == sk1:
            # 一旦有人访问,sk1就会发生变化
            conn, addr = sk.accept()
            conn.sendall(bytes('Hello_World', encoding='utf-8'))
            # 把客户端的socket(conn)方法inputs监听,一旦客户端发送数据过来,conn会发生变化
            inputs.append(conn)
            message_dict[conn] = []
        else:
            # conn发生变化
            try:
                recv_bytes = sk.recv(1024)
            except Exception as e:
                inputs.remove(sk)
                print(e)
            else:
                recv_str = str(recv_bytes, encoding='utf-8')
                message_dict[sk].append(recv_str)
                outputs.append(sk)
            finally:
                pass
    print(w_list)
    for conn in w_list:
        recv_str = message_dict[conn].pop()
        print(recv_str)
        conn.sendall(bytes("回复:" + recv_str, encoding='utf-8'))
    outputs.clear()
梳理
  • select以后基本不会用到,但是这个所有的网络通信的根本,很多源码都会有,如果不懂,则看源码的时候会很吃力
  • socketserver真正实现了并发
    • socket + select + 多线程
  • 多线程
import threading
import time

def process(arg):
    print(arg)
    time.sleep(1)

# 如果这样子执行,会执行10秒    
for i in rang(10):
    process(i)

# 通过多线程,瞬间完成
for i in rang(10):
    t = threading.Thread(target:process,arg=i)
    t.start()
  • socket通信技巧
    • socket在send和recv都有大小限制
    • 如果传输人的数据超过1024,一次接收不完,需要接收多次,如何知道要接收多层次,这就要在发送之前告诉另一端数据有多大
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,185评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,652评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,524评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,339评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,387评论 6 391
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,287评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,130评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,985评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,420评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,617评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,779评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,477评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,088评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,716评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,857评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,876评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,700评论 2 354

推荐阅读更多精彩内容

  • IO操作是不占用CPU的,IO多路复用,是要管理起所有的IO操作。IO多路复用的典型场景是监听socket对象内部...
    极地瑞雪阅读 649评论 0 1
  • Java程序员进阶三条必经之路:数据库、虚拟机、异步通信。 前言 从零单排高性能问题,这次轮到异步通信了。这个领域...
    MountainKing阅读 7,930评论 0 15
  • 上一篇《聊聊同步、异步、阻塞与非阻塞》[//www.greatytc.com/p/aed6067eeac...
    七寸知架构阅读 140,062评论 57 445
  • 本文摘抄自linux基础编程 IO概念 Linux的内核将所有外部设备都可以看做一个文件来操作。那么我们对与外部设...
    VD2012阅读 1,021评论 0 2
  • 《平凡之路》 来一场孤独的遇见, 去拥抱沉默的邂逅, 看着野草葳蕤, 目睹野花枯萎, 它们有怎样的故事, 有人会听...
    林姬阅读 299评论 2 0