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()