美文网首页
03-web服务器v3.1--03

03-web服务器v3.1--03

作者: 努力爬行中的蜗牛 | 来源:发表于2018-11-11 13:25 被阅读4次
    多进程、多线程实现Http服务器
    import socket
    import re
    import multiprocessing
    import threading
    
    
    def service_client(new_socket):
        """为这个客户端返回数据"""
        # 1. 接收浏览器发送过来的请求,即http请求
        # GET / HTTP 1.1
        request = new_socket.recv(1024).decode("utf-8 ")
        # print(">" * 50)
        # print(request)
    
        requet_lines = request.splitlines()
        print("")
        print(">" * 20)
        print(requet_lines)
    
        # GET /index.html HTTP/1.1
        file_name = ""
        # [ ]的^ 表示取反 即不是/ 不是空格
        ret = re.match(r"[^/]+(/[^ ]*)", requet_lines[0])
        print("-" * 100)
        print(ret)
        print("-" * 100)
        if ret:
            file_name = ret.group(1)
            # print("*" * 50, file_name)
            if file_name == "/":
                file_name = "/index.html"
    
        try:
            f = open("./html" + file_name, "rb")
        except:
            response = "HTTP/1.1 404 NOT FOUND\r\n"  # 换行
            response += "\r\n"
            response += "---file not found---"
            new_socket.send(response.encode("utf-8"))
        else:
            html_content = f.read()
            f.close()
            # 2. 返回http格式的数据,给浏览器
            # 2.1 准备发送给浏览器的数据---header
            response = "HTTP/1.1 200 OK\r\n"  # 换行
            response += "\r\n"
            # 2.2 准备发送给浏览器的数据---body
            # response += "<h1>hello world</h1>"
            # new_socket.send(response.encode("utf-8"))
            # 将response header发送给浏览器
            new_socket.send(response.encode("utf-8"))
            # 将response body发送给浏览器
            new_socket.send(html_content)
    
        # 3. 关闭套接字
        new_socket.close()
    
    
    def main():
        # 1. 创建套接字
        tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 设置当服务器先close 即服务器端4次挥手之后资源能够立即释放,这样就保证了下次运行的时候可以立即使用这个地址
        tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1)
        # 2. 绑定
        tcp_server_socket.bind(("127.0.0.1", 7890))
        # 3. 变为监听套接字
        tcp_server_socket.listen(128)
        while True:
            # 4. 等待新客户的链接
            new_socket, client_addr = tcp_server_socket.accept()
            # 开启一条子进程或者线程来服务客户端
            # p = multiprocessing.Process(target=service_client, args=(new_socket, ))
            p = threading.Thread(target=service_client, args=(new_socket, ))
            p.start()
    
            # 多进程需要子进程和主进程同时关闭 才会关闭 fd 文件描述符
            # new_socket.close()
    
    
    if __name__ == '__main__':
        main()
    
    使用gevent实现Http服务器开启多任务
    import socket
    import re
    import gevent
    from gevent import monkey
    
    monkey.patch_all()
    
    
    def service_client(new_socket):
        """为这个客户端返回数据"""
        # 1. 接收浏览器发送过来的请求,即http请求
        # GET / HTTP 1.1
        request = new_socket.recv(1024).decode("utf-8 ")
        # print(">" * 50)
        # print(request)
    
        requet_lines = request.splitlines()
        print("")
        print(">" * 20)
        print(requet_lines)
    
        # GET /index.html HTTP/1.1
        file_name = ""
        # [ ]的^ 表示取反 即不是/ 不是空格
        ret = re.match(r"[^/]+(/[^ ]*)", requet_lines[0])
        print("-" * 100)
        print(ret)
        print("-" * 100)
        if ret:
            file_name = ret.group(1)
            # print("*" * 50, file_name)
            if file_name == "/":
                file_name = "/index.html"
    
        try:
            f = open("./html" + file_name, "rb")
        except:
            response = "HTTP/1.1 404 NOT FOUND\r\n"  # 换行
            response += "\r\n"
            response += "---file not found---"
            new_socket.send(response.encode("utf-8"))
        else:
            html_content = f.read()
            f.close()
            # 2. 返回http格式的数据,给浏览器
            # 2.1 准备发送给浏览器的数据---header
            response = "HTTP/1.1 200 OK\r\n"  # 换行
            response += "\r\n"
            # 2.2 准备发送给浏览器的数据---body
            # response += "<h1>hello world</h1>"
            # new_socket.send(response.encode("utf-8"))
            # 将response header发送给浏览器
            new_socket.send(response.encode("utf-8"))
            # 将response body发送给浏览器
            new_socket.send(html_content)
    
        # 3. 关闭套接字
        new_socket.close()
    
    
    def main():
        # 1. 创建套接字
        tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 设置当服务器先close 即服务器端4次挥手之后资源能够立即释放,这样就保证了下次运行的时候可以立即使用这个地址
        tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1)
        # 2. 绑定
        tcp_server_socket.bind(("127.0.0.1", 7890))
        # 3. 变为监听套接字
        tcp_server_socket.listen(128)
        while True:
            # 4. 等待新客户的链接
            new_socket, client_addr = tcp_server_socket.accept()
            # 使用gevent实现多任务
            gevent.spawn(service_client, new_socket)
    
    
    if __name__ == '__main__':
        main()
    
    单进程、线程、非堵塞实现并发
    import socket
    import time
    
    tcp_server_tcp = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    tcp_server_tcp.bind(("", 7899))
    tcp_server_tcp.listen(128)
    tcp_server_tcp.setblocking(False)  # 设置套接字为非堵塞的方式
    
    client_socket_list = list()
    
    while True:
    
        # time.sleep(0.5)
    
        try:
            new_socket, new_addr = tcp_server_tcp.accept()
        except Exception as ret:
            print("---没有新的客户端到来---")
        else:
            print("---只要没有产生异常,那么也就意味着 来了一个新的客户端----")
            new_socket.setblocking(False)  # 设置套接字为非堵塞的方式
            client_socket_list.append(new_socket)
            
        for client_socket in client_socket_list:
            try:
                recv_data = client_socket.recv(1024)
            except Exception as ret:
                print(ret)
                print("----这个客户端没有发送过来数据----")
            else:
                print("-----没有异常-----")
                print(recv_data)
                if recv_data:
                    # 对方发送过来数据
                    print("----客户端发送过来了数据-----")
                else:
                    # 对方调用close 导致了 recv返回
                    client_socket.close()
                    client_socket_list.remove(client_socket)
                    print("---客户端已经关闭----")
    
    长链接、短连接

    短连接:
    建立连接--数据传输--关闭连接 如此重复
    长连接:
    建立连接--数据传输(连接中...)---数据传输---断开连接

    长连接HTTP服务器

    服务器发送完数据后不会断开连接,服务器会告诉客户端发送数据的长度,客户端接受完数据后才会断开。

    import socket
    import re
    import gevent
    from gevent import monkey
    import time
    
    monkey.patch_all()
    
    
    def service_client(new_socket, request):
        """为这个客户端返回数据"""
        # 1. 接收浏览器发送过来的请求,即http请求
        # GET / HTTP 1.1
        # request = new_socket.recv(1024).decode("utf-8 ")
        # print(">" * 50)
        # print(request)
    
        requet_lines = request.splitlines()
        print("")
        print(">" * 20)
        print(requet_lines)
    
        # GET /index.html HTTP/1.1
        file_name = ""
        # [ ]的^ 表示取反 即不是/ 不是空格
        ret = re.match(r"[^/]+(/[^ ]*)", requet_lines[0])
        print("-" * 100)
        print(ret)
        print("-" * 100)
        if ret:
            file_name = ret.group(1)
            # print("*" * 50, file_name)
            if file_name == "/":
                file_name = "/index.html"
    
        try:
            f = open("./html" + file_name, "rb")
        except:
            response = "HTTP/1.1 404 NOT FOUND\r\n"  # 换行
            response += "\r\n"
            response += "---file not found---"
            new_socket.send(response.encode("utf-8"))
        else:
            html_content = f.read()
            f.close()
    
            # 准备发送给浏览器的数据---body
            response_body = html_content
    
            response_header = "HTTP/1.1 200 OK\r\n"  # 换行
            response_header += "Content-Length:%d\r\n" % len(response_body)  # 告诉浏览器发送的数据长度,浏览器接收完数据才会断开连接
            response_header += "\r\n"
    
            response = response_header.encode("utf-8") + response_body
            # 将response header发送给浏览器
            new_socket.send(response)
    
        # 3. 关闭套接字
        # new_socket.close()
    
    
    def main():
        # 1. 创建套接字
        tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 设置当服务器先close 即服务器端4次挥手之后资源能够立即释放,这样就保证了下次运行的时候可以立即使用这个地址
        tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1)
    
        # 2. 绑定
        tcp_server_socket.bind(("127.0.0.1", 7890))
    
        # 3. 变为监听套接字
        tcp_server_socket.listen(128)
        # 设置套接字为非堵塞方式
        tcp_server_socket.setblocking(False)
        client_socket_list = list()
    
        while True:
            time.sleep(0.5)
            try:
                # 4. 等待新客户的链接
                new_socket, client_addr = tcp_server_socket.accept()
            except Exception as ret:
                pass
            else:
                new_socket.setblocking(False)
                client_socket_list.append(new_socket)
    
            for client_socket in client_socket_list:
                try:
                    recv_data = client_socket.recv(1024).decode("utf-8")
                except Exception as ret:
                    pass
                else:
                    if recv_data:
                        # 对方发送过来数据
                        # 使用gevent实现多任务
                        # gevent.spawn(service_client, client_socket, recv_data.decode("utf-8"))
                        service_client(client_socket, recv_data)
                    else:
                        # 对方调用了close 导致了 recv返回
                        client_socket.close()
                        client_socket_list.remove(client_socket)
        # 关闭监听套接字
        tcp_server_socket.close()
    
    
    if __name__ == '__main__':
        main()
    
    epoll版的Http服务器

    [epoll参考网址]https://blog.csdn.net/xiajun07061225/article/details/9250579

    import socket
    import re
    import select
    
    
    def service_client(new_socket, request):
        """为这个客户端返回数据"""
    
        # 1. 接收浏览器发送过来的请求 ,即http请求  
        # GET / HTTP/1.1
        # .....
        # request = new_socket.recv(1024).decode("utf-8")
        # print(">>>"*50)
        # print(request)
    
        request_lines = request.splitlines()
        print("")
        print(">"*20)
        print(request_lines)
    
        # GET /index.html HTTP/1.1
        # get post put del
        file_name = ""
        ret = re.match(r"[^/]+(/[^ ]*)", request_lines[0])
        if ret:
            file_name = ret.group(1)
            # print("*"*50, file_name)
            if file_name == "/":
                file_name = "/index.html"
    
        # 2. 返回http格式的数据,给浏览器
        
        try:
            f = open("./html" + file_name, "rb")
        except:
            response = "HTTP/1.1 404 NOT FOUND\r\n"
            response += "\r\n"
            response += "------file not found-----"
            new_socket.send(response.encode("utf-8"))
        else:
            html_content = f.read()
            f.close()
    
            response_body = html_content
    
            response_header = "HTTP/1.1 200 OK\r\n"
            response_header += "Content-Length:%d\r\n" % len(response_body)
            response_header += "\r\n"
    
            response = response_header.encode("utf-8") + response_body
    
            new_socket.send(response)
    
    
    def main():
        """用来完成整体的控制"""
        # 1. 创建套接字
        tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
    
        # 2. 绑定
        tcp_server_socket.bind(("", 7890))
    
        # 3. 变为监听套接字
        tcp_server_socket.listen(128)
        tcp_server_socket.setblocking(False)  # 将套接字变为非堵塞
    
        # 创建一个epoll对象
        epl = select.epoll()
    
        # 将监听套接字对应的fd注册到epoll中
        epl.register(tcp_server_socket.fileno(), select.EPOLLIN)
    
        fd_event_dict = dict()
    
        while True:
    
            fd_event_list = epl.poll()  # 默认会堵塞,直到 os监测到数据到来 通过事件通知方式 告诉这个程序,此时才会解堵塞
    
            # [(fd, event), (套接字对应的文件描述符, 这个文件描述符到底是什么事件 例如 可以调用recv接收等)]
            for fd, event in fd_event_list:
                # 等待新客户端的链接
                if fd == tcp_server_socket.fileno():
                    new_socket, client_addr = tcp_server_socket.accept()
                    epl.register(new_socket.fileno(), select.EPOLLIN)
                    fd_event_dict[new_socket.fileno()] = new_socket
                elif event==select.EPOLLIN:
                    # 判断已经链接的客户端是否有数据发送过来
                    recv_data = fd_event_dict[fd].recv(1024).decode("utf-8")
                    if recv_data:
                        service_client(fd_event_dict[fd], recv_data)
                    else:
                        fd_event_dict[fd].close()
                        epl.unregister(fd)
                        del fd_event_dict[fd]
    
    
        # 关闭监听套接字
        tcp_server_socket.close()
    
    
    if __name__ == "__main__":
        main()
    
    
    

    相关文章

      网友评论

          本文标题:03-web服务器v3.1--03

          本文链接:https://www.haomeiwen.com/subject/cuszxqtx.html