概述
可以返回请求的指定页面,没有请求的该文件,则返回404状态信息。
实现步骤
1.导入socket、os包
importsocketimportos
2.创建tcp服务端套接字
tcp_server_socket=socket.socket(socket.AF_INET, socket.SOCK_STREAM)
3.绑定端口号
tcp_server_socket.bind(("", 8000))
4.设置监听
tcp_server_socket.listen(128)
5.等待接受客户端的连接请求
new_socket, ip_port=tcp_server_socket.accept()
6.接收客户端的请求信息
recv_data=new_socket.recv(4096) # 判断接收的数据长度是否为0iflen(recv_data) ==0: new_socket.close() return
7.对二进制数据进行解码
recv_content=recv_data.decode("utf-8") print(recv_content)
8.对数据按照空格进行分割
request_list=recv_content.split(" ", maxsplit=2)
9.获取请求的资源路径
request_path=request_list[1] print(request_path)
10.判断请求的是否是根目录,如果是根目录设置返回的信息
ifrequest_path=="/": request_path="/index.html"
11.报文处理
try: # 打开文件读取文件中的数据, 提示:这里使用rb模式,兼容打开图片文件withopen("static"+request_path, "rb") asfile: # 这里的file表示打开文件的对象file_data=file.read() # 提示: with open 关闭文件这步操作不用程序员来完成,系统帮我们来完成exceptExceptionase: # 代码执行到此,说明没有请求的该文件,返回404状态信息# 响应行response_line="HTTP/1.1 404 Not Found\r\n"# 响应头response_header="Server: PWS/1.0\r\n"# 读取404页面数据withopen("static/error.html", "rb") asfile: file_data=file.read() # 响应体response_body=file_data# 把数据封装成http 响应报文格式的数据response= (response_line+response_header+"\r\n").encode("utf-8") +response_body# 发送给浏览器的响应报文数据new_socket.send(response) else: # 代码执行到此,说明文件存在,返回200状态信息# 响应行response_line="HTTP/1.1 200 OK\r\n"# 响应头response_header="Server: PWS/1.0\r\n"# 响应体response_body=file_data# 把数据封装成http 响应报文格式的数据response= (response_line+response_header+"\r\n").encode("utf-8") +response_body# 发送给浏览器的响应报文数据new_socket.send(response) finally: # 关闭服务于客户端的套接字new_socket.close()
代码实现
importsocketimportosdefmain(): # 创建tcp服务端套接字tcp_server_socket=socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置端口号复用,程序退出端口号立即释放tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) # 绑定端口号tcp_server_socket.bind(("", 8000)) # 设置监听tcp_server_socket.listen(128) # 循环等待接受客户端的连接请求whileTrue: # 等待接受客户端的连接请求new_socket, ip_port=tcp_server_socket.accept() # 代码执行到此,说明连接建立成功# 接收客户端的请求信息recv_data=new_socket.recv(4096) # 判断接收的数据长度是否为0iflen(recv_data) ==0: new_socket.close() return# 对二进制数据进行解码recv_content=recv_data.decode("utf-8") print(recv_content) # 对数据按照空格进行分割request_list=recv_content.split(" ", maxsplit=2) # 获取请求的资源路径request_path=request_list[1] print(request_path) # 判断请求的是否是根目录,如果是根目录设置返回的信息ifrequest_path=="/": request_path="/index.html"# 1. os.path.exits# os.path.exists("static/" + request_path)# 2. try-excepttry: # 打开文件读取文件中的数据, 提示:这里使用rb模式,兼容打开图片文件withopen("static"+request_path, "rb") asfile: # 这里的file表示打开文件的对象file_data=file.read() # 提示: with open 关闭文件这步操作不用程序员来完成,系统帮我们来完成exceptExceptionase: # 代码执行到此,说明没有请求的该文件,返回404状态信息# 响应行response_line="HTTP/1.1 404 Not Found\r\n"# 响应头response_header="Server: PWS/1.0\r\n"# 读取404页面数据withopen("static/error.html", "rb") asfile: file_data=file.read() # 响应体response_body=file_data# 把数据封装成http 响应报文格式的数据response= (response_line+response_header+"\r\n").encode("utf-8") +response_body# 发送给浏览器的响应报文数据new_socket.send(response) else: # 代码执行到此,说明文件存在,返回200状态信息# 响应行response_line="HTTP/1.1 200 OK\r\n"# 响应头response_header="Server: PWS/1.0\r\n"# 响应体response_body=file_data# 把数据封装成http 响应报文格式的数据response= (response_line+response_header+"\r\n").encode("utf-8") +response_body# 发送给浏览器的响应报文数据new_socket.send(response) finally: # 关闭服务于客户端的套接字new_socket.close() # 判断是否是主模块的代码if__name__=='__main__': main()