多任务版本
可以支持多人同时访问
代码实现
import socket
import os
import threading
# 处理客户端请求
def handle_client_request(new_socket):
# 接收客户端的请求信息
recv_data = new_socket.recv(4096)
# 判断如果客户端请求是空数据,则直接关闭服务套接字
if len(recv_data) == 0:
new_socket.close()
return
# 对二进制数据进行解码
recv_content = recv_data.decode("utf-8")
# 对数据按照空格进行分割
request_list = recv_content.split(" ", maxsplit=2)
# 获取请求的资源路径
request_path = request_list[1]
# 打印查看获取到的资源路径
print(request_path)
# 判断请求的是否为根路径,如果是设置默认返回图片
if request_path == "/":
request_path = "/test.jpeg"
# 判断此文件是否存在
# 第一种方法:os.path.exists
# os.path.exists("web/" + request_path)
# 第二种方法:try-except
try:
# 准备返回数据
# 打开文件读取文件中的数据,with open关闭文件无需程序员操作,系统自动关闭
# 这里使用"rb"模式,兼容打开图片文件
with open("web" + request_path, "rb") as file: # file表示打开文件的对象
file_data = file.read()
except Exception as e:
# 代码执行到此,说明没有请求的文件资源,返回404状态信息
# 响应行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 响应头
response_header = "Server: LAIFU/1.0\r\n"
# 读取404页面数据
with open("web/error.html", "rb") as file:
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: LAIFU/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()
def main():
# 创建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)
# 循环等待接收客户端的连接请求
while True:
new_socket, ip_port = tcp_server_socket.accept()
sub_thread = threading.Thread(target=handle_client_request, args=(new_socket,))
# 设置成为守护主线程
sub_thread.setDaemon(True)
# 启动子线程执行对应的任务
sub_thread.start()
# 判断是否是主模块的代码
if __name__ == '__main__':
main()
面向对象开发
代码实现
import socket
import os
import threading
# http协议的web服务器类
class HttpWebServer(object):
def __init__(self):
# 创建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)
# 把tcp服务器的套接字作为web服务器对象的属性
self.tcp_server_socket = tcp_server_socket
# 处理客户端请求
@staticmethod
def handle_client_request(new_socket):
# 接收客户端的请求信息
recv_data = new_socket.recv(4096)
# 判断如果客户端请求是空数据,则直接关闭服务套接字
if len(recv_data) == 0:
new_socket.close()
return
# 对二进制数据进行解码
recv_content = recv_data.decode("utf-8")
# 对数据按照空格进行分割
request_list = recv_content.split(" ", maxsplit=2)
# 获取请求的资源路径
request_path = request_list[1]
# 打印查看获取到的资源路径
print(request_path)
# 判断请求的是否为根路径,如果是设置默认返回图片
if request_path == "/":
request_path = "/test.jpeg"
# 判断此文件是否存在
# 第一种方法:os.path.exists
# os.path.exists("web/" + request_path)
# 第二种方法:try-except
try:
# 准备返回数据
# 打开文件读取文件中的数据,with open关闭文件无需程序员操作,系统自动关闭
# 这里使用"rb"模式,兼容打开图片文件
with open("web" + request_path, "rb") as file: # file表示打开文件的对象
file_data = file.read()
except Exception as e:
# 代码执行到此,说明没有请求的文件资源,返回404状态信息
# 响应行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 响应头
response_header = "Server: LAIFU/1.0\r\n"
# 读取404页面数据
with open("web/error.html", "rb") as file:
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: LAIFU/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()
# 启动服务器的方法
def start(self):
# 循环等待接收客户端的连接请求
while True:
new_socket, ip_port = self.tcp_server_socket.accept()
sub_thread = threading.Thread(target=self.handle_client_request, args=(new_socket,))
# 设置成为守护主线程
sub_thread.setDaemon(True)
# 启动子线程执行对应的任务
sub_thread.start()
def main():
# 创建web服务器
web_server = HttpWebServer()
# 启动服务器
web_server.start()
# 判断是否是主模块的代码
if __name__ == '__main__':
main()
获取终端命令行参数
代码
import sys
# 获取终端命令行参数
params = sys.argv
print(params, type(params))
效果展示
在终端进行命令执行 python3 myFirstPython.py 9000
小结:
结果类型为列表,列表里的每项数据都是字符串类型
命令行启动动态绑定端口号
代码实现
命令行动态输入端口号绑定启动程序,限制输入两个参数,第二个参数为int类型
import socket
import os
import threading
import sys
# http协议的web服务器类
class HttpWebServer(object):
def __init__(self, port):
# 创建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(("", port))
# 配置监听
tcp_server_socket.listen(128)
# 把tcp服务器的套接字作为web服务器对象的属性
self.tcp_server_socket = tcp_server_socket
# 处理客户端请求
@staticmethod
def handle_client_request(new_socket):
# 接收客户端的请求信息
recv_data = new_socket.recv(4096)
# 判断如果客户端请求是空数据,则直接关闭服务套接字
if len(recv_data) == 0:
new_socket.close()
return
# 对二进制数据进行解码
recv_content = recv_data.decode("utf-8")
# 对数据按照空格进行分割
request_list = recv_content.split(" ", maxsplit=2)
# 获取请求的资源路径
request_path = request_list[1]
# 打印查看获取到的资源路径
print(request_path)
# 判断请求的是否为根路径,如果是设置默认返回图片
if request_path == "/":
request_path = "/test.jpeg"
# 判断此文件是否存在
# 第一种方法:os.path.exists
# os.path.exists("web/" + request_path)
# 第二种方法:try-except
try:
# 准备返回数据
# 打开文件读取文件中的数据,with open关闭文件无需程序员操作,系统自动关闭
# 这里使用"rb"模式,兼容打开图片文件
with open("web" + request_path, "rb") as file: # file表示打开文件的对象
file_data = file.read()
except Exception as e:
# 代码执行到此,说明没有请求的文件资源,返回404状态信息
# 响应行
response_line = "HTTP/1.1 404 Not Found\r\n"
# 响应头
response_header = "Server: LAIFU/1.0\r\n"
# 读取404页面数据
with open("web/error.html", "rb") as file:
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: LAIFU/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()
# 启动服务器的方法
def start(self):
# 循环等待接收客户端的连接请求
while True:
new_socket, ip_port = self.tcp_server_socket.accept()
sub_thread = threading.Thread(target=self.handle_client_request, args=(new_socket,))
# 设置成为守护主线程
sub_thread.setDaemon(True)
# 启动子线程执行对应的任务
sub_thread.start()
def main():
# 获取终端命令行参数
params = sys.argv
# 判断传入的参数为两个,并且第二个参数为数字组成的字符串
if len(params) != 2:
print("请执行命令格式如下:python3 文件名 端口号")
print("例如:python3 xxx.py 9000")
return
if not params[1].isdigit():
print("请执行命令格式如下:python3 文件名 端口号")
print("例如:python3 xxx.py 9000")
return
# 代码执行至此,说明输入的参数格式正确,将输入的端口转为int类型
port = int(params[1])
# 创建web服务器
web_server = HttpWebServer(port)
# 启动服务器
web_server.start()
# 判断是否是主模块的代码
if __name__ == '__main__':
main()
效果展示
1、参数输多或第二参数不是int类型
2、参数正确正常启动,浏览器访问没问题