zoukankan      html  css  js  c++  java
  • IO阻塞模型、IO非阻塞模型、多路复用IO模型

    IO操作主要包括两类:

    • 本地IO

    • 网络IO

    本地IO:本地IO是指本地的文件读取等操作,本地IO的优化主要是在操作系统中进行,我们对于本地IO的优化作用十分有限

    网络IO:网络IO指的是在进行网络操作时需要等待用户的输入及传输的等待等,网络IO的优化需要我们自己进行,而我们对于网络IO的优化主要在等待用户输入时程序可以继续运行

    1、IO阻塞模型

    什么是IO阻塞模型

    在我们使用socket创建客户端、服务端时,如果不对 他们执行其他操作,那么客户端的recv、send和服务器端的accept、send、recv等都是阻塞的,只有等到有数据传输过来或者有客户端连接过来时才会操作,否则就会进入等待状态,这种模型就是IO阻塞模型

    IO阻塞模型的缺点

    使用IO阻塞模型时,客户端的影响较小,但是对于服务器端,由于要处理多个客户端的请求,所以如果使用阻塞模型,那么同一时间只能有一个客户端进行连接,效率十分低,不能进行并发

    2、IO非阻塞模型

    什么是IO非阻塞模型

    由于在使用网络IO时,在不进行任何处理的情况下默认是会阻塞的,但是如果不想程序进行阻塞,此时可以通过设置setblocking来实现,这样在进行原本会阻塞的操作时,如果有数据就会对数据进行处理,如果没有数据则会直接报错,只要进行异常的捕捉就能使程序进行后续代码的执行,这样可以实现IO非阻塞

    示例代码:

    客户端

    import socket
    ​
    client = socket.socket()
    client.connect(("127.0.0.1",1688))
    ​
    while True:
        msg = input("msg:").strip()
        if not msg:
            continue
        try:
            client.send(msg.encode("utf-8"))
            recv_msg = client.recv(2048)
            print(recv_msg.decode("utf-8"))
        except ConnectionResetError:
            print("客户端意外关闭")
            break

    服务器

    import socket
    ​
    server = socket.socket()
    server.bind(("127.0.0.1",1688))
    server.listen()
    server.setblocking(False)
    ​
    conn_list = []
    msg_list = []
    ​
    while True:
        try:
            conn,addr = server.accept()
            conn_list.append(conn)
        except BlockingIOError:
            print("还没有客户端连接")
            for conn in conn_list:
                try:
                    msg = conn.recv(1024)
                    msg_list.append((conn,msg))
                except BlockingIOError:
                    print("该用户没有数据传输过来")
            for msg_t in msg_list:
                conn,msg = msg_t
                try:
                    conn.send(msg.upper())
                    msg_list.remove(msg_t)
                except ConnectionResetError:
                    print("信息无法发送")
                    conn.close()
                    conn_list.remove(conn)

                   

    IO非阻塞模型的缺点

    使用IO非阻塞模型,我们解决了不能实现并发的缺点,在一个线程中实现了并发,但是IO非阻塞模型存在一些问题,最主要的问题是,在使用非阻塞模型时,由于需要不停的进行询问,所以会持续的消耗系统的CPU资源,造成不必要的CPU占用

    3、多路复用IO模型

    在使用非阻塞IO模型处理问题时,虽然解决了不能在单个线程中实现并发的问题,但是由于需要不停的进行询问,所以就会造成CPU的不必要占用,造成CPU占用过高的问题

    什么是多路复用IO模型

    多路复用IO模型指的是多个TCP连接使用一个或者少量的线程来实现通信的IO模型,在IO非阻塞模型中,我们是通过自己不停的使用send()或者recv()来不停的询问是否有数据需要进行操作,在多路复用IO模型中,我们使用select统一的来进行询问,并将可以进行操作的对象放到一个列表中进行统一管理,并且select还可以区分那些是可以发送数据的对象,哪些是可以接收数据的对象,并放在不同的列表中进行统一管理

    示例代码:

    客户端

    import socket
    ​
    client = socket.socket()
    client.connect(("127.0.0.1",1688))
    ​
    while True:
        try:
            msg = input("msg:").strip()
            if not msg:
                continue
            client.send(msg.encode("utf-8"))
            recv_msg = client.recv(1024).decode("utf-8")
            print(recv_msg)
        except ConnectionResetError:
            print("服务器已关闭")
            break

    服务器端

    import socket
    import select
    ​
    server = socket.socket()
    server.bind(("127.0.0.1",1688))
    server.listen()
    ​
    r_list = [server,]
    w_list = []
    msg_list = []
    while True:
        readable_list,writeable_list,_ = select.select(r_list,w_list,[])
        for conn in readable_list:
            if conn == server:
                conn,addr = conn.accept()
                r_list.append(conn)
            else:
                try:
                    msg = conn.recv(1024)
                    if not msg:
                        raise ConnectionResetError
                    msg_list.append((conn,msg))
                    w_list.append(conn)
                except ConnectionResetError:
                    print("%s客户端正常关闭" %conn)
                    r_list.remove(conn)
                    if conn in w_list:
                        w_list.remove(conn)
                    conn.close()
        for conn in w_list:
            for msg_t in msg_lis[:]:
                connection,msg = msg_t
                if conn == connection:
                    try:
                        connection.send(msg.upper())
                msg_list.remove(msg_t)
    except ConnectionResetError: msg_list.remove(msg_t) w_list.remove(conn) ​ msg_list.remove(msg_t) w_list.remove(conn)
     

    多路复用的缺点:

    使用多路复用解决了在非阻塞IO中出现的CPU占用过高的问题,但是在多路复用中也出现了几个问题:

    • 使用select时最多只能处理1024个客户端,如果数量多与此值,那么就会直接就会报错

    • 如果发送的数据量特别大的情况下只能处理一个客户端,其它的客户端只能进行等待

  • 相关阅读:
    三种负载均衡 Nginx、Dubbo、Ribbon 区别
    Docker基础学习
    主从复制报错2061:Authentication plugin 'caching_sha2_password' reported error:Authentication require secure connection
    LRU、LFU算法区别
    CAP理论原理
    Mysql安装服务到Window服务列表
    从零开始掌握 HAProxy 负载均衡器,详细!!
    一举拿下Nginx
    Nginx 负载均衡配置误区
    Linux自动化技巧
  • 原文地址:https://www.cnblogs.com/lice-blog/p/11000908.html
Copyright © 2011-2022 走看看