欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Web服务器

程序员文章站 2022-05-08 23:53:07
...

Web服务器——单线程单进程非阻塞实现高并发长连接

1.非阻塞:在套接字accept的时候,如果没有链接,则会一直阻塞在那里,这样就不会执行下面的服务逻辑,也就是说,而解阻塞之后,不会产生阻塞,可以流畅的进行服务。这里我们需要建立一个服务套接字的列表,在没有收到新的服务器连接时,因为不阻塞,也能取列表的服务套接字进行服务。
2.长连接与短链接:
http 1.1:长连接:三次握手后,重复数据多次传输,四次挥手断开
http 1.0:短链接:每一次数据传输都要进行一次握手挥手
短链接:传输短时间少数据,用于简洁的网站,之前的内容中在发数据完后我们都close 了,虽然用的是HTTP1.1但实质上我们强制关闭变成了短链接
长连接:传输时间长大量数据,如王者荣耀服务器
现在多用的是长连接,这就需要我们返回关键字段:“Content-Length:%d\r\n” %len(content)表示返回数据的长度,在达到长度之后,客户端停止接收数据,可以直接发送新的请求给服务器,而不用断开连接再重新建立,再发请求


import socket
import re
import threading
def getname(ask):
    name=ask.splitlines()
    #print(name)
    ret=re.match(r"[^/]+([^ ]*)",name[0])
    if ret:
        filename=ret.group(1)
        if filename=="/":
            filename="/Music.html"
    return filename
def sendcontent(filename,severwindow):
    try:
        with open(".\\" + filename, 'rb') as f:
            content = f.read()
    except:
        data = "HTTP/1.1 404 NOT FOUND\r\n"
        data += "\r\n"
        severwindow.send(data.encode("utf-8"))
    else:
        data = "HTTP/1.1 200 OK\r\n"
        data +="Content-Length:%d\r\n" %len(content)#标记数据长度,长连接的关键
        data += "\r\n"
        severwindow.send(data.encode("utf-8"))
        severwindow.send(content)

def serve(severwindow,window):
    #接收并打印请求
    try:
        request=severwindow.recv(1024).decode("utf-8")
    except Exception as f:
        print("--未接收到客户端数据")
    else:
        if request:
            print(request)
            #提取request请求
            filename=getname(request)
            #返回页面
            #发送文件页面内容:
            sendcontent(filename,severwindow)
            window.remove(severwindow)#服务完毕,删除套接字
            #severwindow.close()#实际上这里造成了短链接
            #浏览器收数据,但不知道收没收完的情况下,不会再发一次请求,这时候就需要回服务器告诉浏览器发的数据有多少,浏览器接收够了就发起下一次请求
        else:
            severwindow.close()#长连接情况下,只会发生客户端率先关闭链接,则recv为空,这时候服务器再关链接
def main():
    #创建套接字
    sever=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    #绑定
    sever.bind(("",8080))
    #监听
    sever.listen(128)
    sever.setblocking(False)#非阻塞的套接字
    window=list()#服务套接字列表
    while True:
        #接收
        try:
            severwindow,client_addr=sever.accept()
        except Exception as ret:
            print("--无客户端到来--")
        else:
            severwindow.setblocking(False)
            window.append(severwindow)
            for client in window:
                serve(client,window)
    sever.close()
if __name__ == '__main__':
    main()