HTTP协议和python实现静态Web服务器

一、网址URL

URL 就是网络资源地址,也就是我们常说的网址。

URL的样子:

https://fanyi.youdao.com/?keyfrom=dict2.index

URL的组成部分:

  1. 协议部分: https://、http://、ftp://

    协议:即双方都遵循的数据传输协议

  2. 域名部分: fanyi.youdao.com

    域名:IP地址的别名,由一串用点分隔的名字组成的Internet上某一台计算机或计算机组的名称,用于在数据传输时对计算机的定位标识(有时也指地理位置)。

    由于IP地址具有不方便记忆并且不能显示地址组织的名称和性质等缺点,人们设计出了域名,并通过网域名称系统(DNS,Domain Name System)来将域名和IP地址相互映射,使人更方便地访问互联网,而不用去记住能够被机器直接读取的IP地址数串。

  3. 资源路径部分:

    /?keyfrom=dict2.index

    ?后是查询参数部分,keyfrom是第一个资源,若还有资源则用&进行连接

二、HTTP协议

1. HTTP 协议的介绍

HTTP 协议是一个超文本传输协议,它是浏览器Web服务器传输数据的一个协议。传输 HTTP 协议格式的数据是基于 TCP 传输协议的,发送数据之前需要先建立连接。

  • TCP传输协议是用来保证网络中传输的数据的安全性的
  • HTTP协议是用来规定这些数据的具体格式的。

超文本是超级文本的缩写,是指超越文本限制或者超链接,比如:图片、音乐、视频、超链接等等都属于超文本。

(2)浏览器访问web服务器的通信过程

访问web服务器的通信过程

2. HTTP请求报文
(1)HTTP最常见的请求报文有两种:
  1. GET 方式的请求报文:获取web服务器数据
  2. POST 方式的请求报文:向web服务器提交数据
(2)GET 请求报文说明:

组成部分:

  1. 请求行 格式:GET请求方式 请求资源路径 HTTP协议版本
  2. 请求头 格式:key值:value值
  3. 空行

注每行(即每项数据)后面都需要有\r\n,并且在请求头信息后面还有一个单独的\r\n(这个是空行的)

---- 请求行 ----
GET / HTTP/1.1  # GET请求方式 请求资源路径 HTTP协议版本(之间用空格分开)
---- 请求头 ----- 
Host: www.itcast.cn  # 服务器的主机地址和端口号,默认是80
Connection: keep-alive # 和服务端保持长连接,若浏览器中一直和服务器保存连接则叫长连接;若断开比如看小说的时候时间很长,服务器断开,当换章的时候再和服务器连接,这叫短链接
Upgrade-Insecure-Requests: 1 # 让浏览器升级不安全请求,使用https请求 ,若是0则是不安全
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.100 Safari/537.36  # 用户代理,也就是客户端的名称
Accept:text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8 # 可接受的数据类型,q表示一个权值,q越大表示优先级越靠前,若没有则表示最优先的
Accept-Encoding: gzip, deflate # 可接受的压缩格式
Accept-Language: zh-CN,zh;q=0.9 #可接受的语言
Cookie: pgv_pvi=1246921728; # 登录用户的身份标识

---- 空行 ----

GET 请求原始报文说明:

GET / HTTP/1.1\r\n
Host: www.itcast.cn\r\n  
Connection: keep-alive\r\n
Upgrade-Insecure-Requests: 1\r\n
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.100 Safari/537.36\r\n
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8\r\n
Accept-Encoding: gzip, deflate\r\n
Accept-Language: zh-CN,zh;q=0.9\r\n
Cookie: pgv_pvi=1246921728; \r\n
\r\n  (请求头信息后面还有一个单独的’\r\n’不能省略)
(3)POST 请求报文说明:

组成部分:

  1. 请求行 格式:GET请求方式 请求资源路径 HTTP协议版本

  2. 请求头 格式:key值:value值

  3. 空行

  4. 请求体,请求的具体数据**(这是GET方式没有的)**

    格式:key1=value1&key2=value2

注意:POST方式也允许没有请求体,但很少见;响应体不需要\r\n

---- 请求行 ----
POST /xmweb?host=mail.itcast.cn&_t=1542884567319 HTTP/1.1 # POST请求方式 请求资源路径 HTTP协议版本
---- 请求头 ----
Host: mail.itcast.cn # 服务器的主机地址和端口号,默认是80
Connection: keep-alive # 和服务端保持长连接
Content-Type: application/x-www-form-urlencoded  # 告诉服务端请求的数据类型
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.100 Safari/537.36 # 客户端的名称
---- 空行 ----
---- 请求体 ----
username=hello&pass=hello # 请求参数

POST 请求原始报文说明:

POST /xmweb?host=mail.itcast.cn&_t=1542884567319 HTTP/1.1\r\n
Host: mail.itcast.cn\r\n
Connection: keep-alive\r\n
Content-Type: application/x-www-form-urlencoded\r\n
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.100 Safari/537.36\r\n
\r\n(请求头信息后面还有一个单独的’\r\n’不能省略)
username=hello&pass=hello
3. HTTP相应报文

组成部分:

  1. 相应行 格式:HTTP协议版本 状态码 状态描述本
  2. 相应头 格式:key值:value值
  3. 空行
  4. 相应体 格式:HTTP协议版本 状态码 状态描述,最常见的状态码是200

注意:

  • 响应体不需要\r\n
  • 状态码:
    • 200:服务器已成功处理了请求
    • 400:错误的请求,请求地址或参数有误
    • 404:请求资源在服务器不存在
    • 500:服务器内部源代码出现错误

响应报文说明:

--- 响应行/状态行 ---
HTTP/1.1 200 OK # HTTP协议版本 状态码 状态描述
--- 响应头 ---
Server: Tengine # 服务器名称
Content-Type: text/html; charset=UTF-8 # 内容类型
Transfer-Encoding: chunked # 发送给客户端内容不确定内容长度,发送结束的标记是0\r\n, Content-Length表示服务端确定发送给客户端的内容大小,但是二者只能用其一。
Connection: keep-alive # 和客户端保持长连接
Date: Fri, 23 Nov 2018 02:01:05 GMT # 服务端的响应时间
--- 空行 ---
--- 响应体 ---
<!DOCTYPE html><html lang=“en”> …</html> # 响应给客户端的数据

原始响应报文说明:

HTTP/1.1 200 OK\r\n
Server: Tengine\r\n
Content-Type: text/html; charset=UTF-8\r\n
Transfer-Encoding: chunked\r\n
Connection: keep-alive\r\n
Date: Fri, 23 Nov 2018 02:01:05 GMT\r\n
\r\n(响应头信息后面还有一个单独的’\r\n’不能省略)
<!DOCTYPE html><html lang=“en”> …</html>
4. HTTP协议的通讯过程
(1)谷歌浏览器开发者工具的使用

谷歌浏览器的开发者工具是查看http协议的通信过程利器,通过Network标签选项可以查看每一次的请求和响应的通信过程,调出开发者工具的通用方法是在网页右击选择检查。

首先需要安装Google Chrome浏览器,然后Windows和Linux平台按F12调出开发者工具, mac OS选择 视图 -> 开发者 -> 开发者工具或者直接使用 alt+command+i 这个快捷键,还有一个多平台通用的操作就是在网页右击选择检查
谷歌浏览器开发者工具

(2)开发者工具的标签选项说明:
  • 元素(Elements):用于查看或修改HTML标签
  • 控制台(Console):执行js代码
  • 源代码(Sources):查看静态资源文件,断点调试JS代码
  • 网络(Network):查看http协议的通信过程
(3)开发者工具的使用说明:
  1. 点击Network标签选项
  2. 在浏览器的地址栏输入百度的网址,就能看到请求百度首页的http的通信过程
  3. 这里的每项记录都是请求+响应的一次过程
(4)开发者工具的Headers(标头)选项总共有三部分组成:

谷歌浏览器开发者工具-标头1
谷歌浏览器开发者工具-标头2

  1. General(常规): 主要信息
  2. Response Headers(响应标头): 响应头
  3. Request Headers(请求标头): 请求头

Response(响应)选项是查看响应体信息的

在这里插入图片描述
注意:每一次浏览器和服务器的数据通讯,都是成对出现的请求和响应,同时每一次请求和响应都必须符合HTTP协议的格式

三、静态Web服务器

1. 搭建Python自带的静态Web服务器
(1)静态Web服务器是什么?

可以为发出请求的浏览器提供静态文档的程序

平时我们浏览新闻数据的时候,每天的新闻数据都会发生变化,那访问的这个页面就是动态的,而我们开发的是静态的,页面的数据不会发生变化

静态页面的编码是html文件,最好对前端HTML有一些了解

(2)搭建Python自带的静态Web服务器

搭建Python自带的静态Web服务器使用(在终端中)

1.进入你自己的静态文件目录
cd Desktop/Webtest/
2.开始搭建
python3 -m http.server 端口号
注:不写端口号默认8

-m表示运行包里面的模块,执行这个命令的时候,需要进入你自己指定静态文件的目录,然后通过浏览器就能访问对应的 html文件了,这样一个静态的web服务器就搭建好了。

通过浏览器访问搭建的静态Web服务器,地址为终端给出的端口号地址,eg:http://0.0.0.0:8000/

退出在终端按ctrl+c

进入http://0.0.0.0:端口号/时可以通过谷歌开发者工具查看Headers标头信息和response响应体

2. 返回固定页面数据
(1)开发自己的静态Web服务器

实现步骤:

  1. 编写一个TCP服务端程序
  2. 获取浏览器发送的http请求报文数据
  3. 读取固定页面数据,把页面数据组装成HTTP响应报文数据发送给浏览器。
  4. HTTP响应报文数据发送完成以后,关闭服务于客户端的套接字。
(2)静态Web服务器-返回固定页面数据的示例代码
import socket

if __name__=='__main__':
    # 1.编写一个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(("",8080))
    # 设置监听
    tcp_server_socket.listen(128)
    while True:
        # 2.获取浏览器发送的HTTP请求报文数据
        # 建立连接,等待接收客户端连接请求
        server_client_socket,client_addr = tcp_server_socket.accept()
        # 获取浏览器的请求信息
        client_request_data=server_client_socket.recv(1024).decode()
        print(client_request_data)

        # 3.读取固定页面数据,把页面数据组装成HTTP响应报文数据发送给浏览器
        with open("./Web/index.html","rb") as f: # 以二进制可读的方式进行读取
            file_data=f.read()
        # 应答行
        response_line="HTTP/1.1 200 OK\r\n"
        # 应答头,自己设置名称
        response_header="Server:PythonWeb\r\n"
        # 应答体,如html代码数据
        response_body=file_data

        # 整理整体的应答数据,记住空行的”\r\n“以及因为response_body已经是二进制文件,前面的应答行和应答头以及空行的"\r\n"也需要转化成二进制
        response_data=(response_line+response_header+"\r\n").encode()+response_body
        # 传输数据
        server_client_socket.send(response_data)

        # 4.HTTP响应报文数据发送完成以后,关闭服务于客户端的套接字
        server_client_socket.close()
3. 返回指定页面数据

返回固定页面数据时,只会返回那一个页面,哪怕你访问index2.html,也会返回index.html,这显然是不可以的,所以需要通过用户请求而访问指定的页面数据

读取固定页面是因为代码中

 with open("./Web/index.html","rb") as f: # 以二进制可读的方式进行读取
            file_data=f.read()

只会读取这个文件

(1)返回指定页面数据的实现步骤:
  1. 获取用户请求资源的路径
request_data=client_request_data.split(" ")
request_path=request_data[1] #获取用户请求资源的路径
# 若直接访问网址,不加请求文件,则请求路径为'/',即根目录,这时会报错,所以需要设置成若直接访问网址,默认访问index.html
if request_path =="/":
    request_path="/index.html"

GET /index.html HTTP/1.1

因为请求信息是以空格分开的,只要读取第二个空格即为请求路径

注意:若直接访问网址,不加请求文件,则请求路径为’/',即根目录,这时会报错,所以需要设置成若直接访问网址,默认访问index.html

  1. 根据请求资源的路径,读取指定文件的数据
with open("./Web"+request_path,"rb") as f: # 以二进制可读的方式进行读取
    file_data=f.read()

通过组装打开指定文件。

  1. 组装指定文件数据的响应报文,发送给浏览器

和返回固定页面数据相同

  1. 判断请求的文件在服务端不存在,组装404状态的响应报文,发送给浏览器
 try:
     # 打开指定文件,代码省略...
 except Exception as e:
     server_client_socket.send(404响应报文数据)

通过try进行测试,若有错误则返回404,否则正常进行

(2)静态Web服务器-返回指定页面数据的示例代码
import socket

if __name__ == '__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(("", 8080))
    # 设置监听
    tcp_server_socket.listen(128)
    while True:
        # 等待接受客户端的连接请求
        server_client_socket, client_addr = tcp_server_socket.accept()
        # 代码执行到此,说明连接建立成功
        client_request_data = server_client_socket.recv(4096)

        # 对二进制数据进行解码
        server_client_socket = client_request_data.decode()
        print(server_client_socket)
        # 根据指定字符串进行分割
        request_list = server_client_socket.split(" ")

        # 获取请求资源路径
        request_path = request_list[1]
        print(request_path)

        # 判断请求的是否是根目录,如果条件成立,指定首页数据返回
        if request_path == "/":
            request_path = "/index.html"

        try:
            # 动态打开指定文件
            with open("./Web"+request_path, "rb") as file:
                # 读取文件数据
                file_data = file.read()
        except Exception as e:
            # 请求资源不存在,返回404数据
            # 响应行
            response_line = "HTTP/1.1 404 Not Found\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            with open("./Web/error.html", "rb") as file:
                file_data = file.read()
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        else: #正常返回数据
            # 响应行
            response_line = "HTTP/1.1 200 OK\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        finally:
            # 关闭服务与客户端的套接字
            server_client_socket.close()

(3)注意:关闭客户端,程序会报错(超出列表范围)

原因是因为浏览器和服务器进行通讯,如果浏览器关闭,浏览器会给服务器发一个信息告知,而发送的信息是一个空字符串。但是服务器还是会去解析它,通过

# 根据指定字符串进行分割
request_list = client_request_data.split(" ")
# 获取请求资源路径
request_path = request_list[1]

这一步时,因为信息是空字符串,所以分割后request_list[1]超出列表范围,会报错,所以可以在request_path = request_list[1]前加一个判断:

# 判断客户端是否关闭:
    # 若返回的数据只包含一个,则表示返回的是浏览器返回的空字符串信息告知,则表示浏览器关闭,这时直接关闭套接字就可,并直接结束函数
    if len(request_list) == 1:
        server_client_socket.close()
        return
```
4. 多任务版

目前的Web服务器,不能支持多用户同时访问,只能一个一个的处理客户端的请求,那么如何开发多任务版的web服务器同时处理多个客户端的请求?

可以使用多线程,比进程更加节省内存资源。

(1)多任务版web服务器程序的实现步骤:
  1. 当客户端和服务端建立连接成功,创建子线程,使用子线程专门处理客户端的请求,防止主线程阻塞。
  2. 把创建的子线程设置成为守护主线程,防止主线程无法退出。
(2)静态Web服务器-多任务版的示例代码
import socket
import threading

def handle_client_request(server_client_socket):
    # 获取浏览器的请求信息
    client_request_data = server_client_socket.recv(4096)
    # 对二进制数据进行解码
    client_request_data = client_request_data.decode()
    print(client_request_data)
    # 根据指定字符串进行分割
    request_list = client_request_data.split(" ")
    # 判断客户端是否关闭:
    # 若返回的数据只包含一个,则表示返回的是浏览器返回的空字符串信息告知,则表示浏览器关闭,这时直接关闭套接字就可,并直接结束函数
    if len(request_list) == 1:
        server_client_socket.close()
        return
    # 获取请求资源路径
    request_path = request_list[1]
    print(request_path)
    # 判断请求的是否是根目录,如果条件成立,指定首页数据返回
    if request_path == "/":
        request_path = "/index.html"
    try:
        # 动态打开指定文件
        with open("./Web" + request_path, "rb") as file:
            # 读取文件数据
            file_data = file.read()
    except Exception as e:
        # 请求资源不存在,返回404数据
        # 响应行
        response_line = "HTTP/1.1 404 Not Found\r\n"
        # 响应头
        response_header = "Server: PythonWeb\r\n"
        with open("./Web/error.html", "rb") as file:
            file_data = file.read()
        # 响应体
        response_body = file_data
        # 拼接响应报文
        response_data = (response_line + response_header + "\r\n").encode() + response_body
        # 发送数据
        server_client_socket.send(response_data)
    else:  # 正常返回数据
        # 响应行
        response_line = "HTTP/1.1 200 OK\r\n"
        # 响应头
        response_header = "Server: PythonWeb\r\n"
        # 响应体
        response_body = file_data
        # 拼接响应报文
        response_data = (response_line + response_header + "\r\n").encode() + response_body
        # 发送数据
        server_client_socket.send(response_data)
    finally:
        # 关闭服务与客户端的套接字
        server_client_socket.close()

if __name__ == '__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(("", 8080))
    # 设置监听
    tcp_server_socket.listen(128)
    while True:
        # 等待接受客户端的连接请求
        server_client_socket, client_addr = tcp_server_socket.accept()
        # 代码执行到此,说明连接建立成功
        # 创建子线程,传递参数是为了和客户端进行通信的,即传递server_client_socket
        sub_thread=threading.Thread(target=handle_client_request,args=(server_client_socket,))
        # 启动子线程
        sub_thread.start()
5. 面向对象开发
(1) 以面向对象的方式开发静态Web服务器实现步骤:
  1. 把提供服务的Web服务器抽象成一个类(HTTPWebServer)
  2. 提供Web服务器的初始化方法,在初始化方法里面创建socket对象
  3. 提供一个开启Web服务器的方法,让Web服务器处理客户端请求操作。
(2)以面向对象的方式开发静态Web服务器示例代码
import socket
import threading

# 定义web服务器类
class HttpWebServer(object):
    def __init__(self):
        # 创建tcp服务端套接字
        self.tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 设置端口号复用, 程序退出端口立即释放
        self.tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
        # 绑定端口号
        self.tcp_server_socket.bind(("", 8080))
        # 设置监听
        self.tcp_server_socket.listen(128)

    # 处理客户端的请求,放start函数前
    @staticmethod
    def handle_client_request(self,server_client_socket):
        # 获取浏览器的请求信息
        client_request_data = server_client_socket.recv(4096)
        # 对二进制数据进行解码
        client_request_data = client_request_data.decode()
        print(client_request_data)
        # 根据指定字符串进行分割
        request_list = client_request_data.split(" ")
        # 判断客户端是否关闭:
        # 若返回的数据只包含一个,则表示返回的是浏览器返回的空字符串信息告知,则表示浏览器关闭,这时直接关闭套接字就可,并直接结束函数
        if len(request_list) == 1:
            server_client_socket.close()
            return
        # 获取请求资源路径
        request_path = request_list[1]
        print(request_path)
        # 判断请求的是否是根目录,如果条件成立,指定首页数据返回
        if request_path == "/":
            request_path = "/index.html"
        try:
            # 动态打开指定文件
            with open("./Web" + request_path, "rb") as file:
                # 读取文件数据
                file_data = file.read()
        except Exception as e:
            # 请求资源不存在,返回404数据
            # 响应行
            response_line = "HTTP/1.1 404 Not Found\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            with open("./Web/error.html", "rb") as file:
                file_data = file.read()
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        else:  # 正常返回数据
            # 响应行
            response_line = "HTTP/1.1 200 OK\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        finally:
            # 关闭服务与客户端的套接字
            server_client_socket.close()

    # 启动Web服务器进行工作
    def start(self):
        while True:
            # 等待接受客户端的连接请求
            server_client_socket, client_addr = self.tcp_server_socket.accept()
            # 代码执行到此,说明连接建立成功
            # 创建子线程,传递参数是为了和客户端进行通信的,即传递server_client_socket,同时设置守护主线程
            sub_thread=threading.Thread(target=self.handle_client_request,args=(server_client_socket,),daemon=True)
            # 启动子线程
            sub_thread.start()

if __name__=='__main__':
    # 创建web服务器对象
    web_server=HttpWebServer()
    # 启动Web服务器进行工作
    web_server.start()
6. 命令行启动Web静态服务器动态绑定端口号
(1)动态绑定端口号的静态web服务器

若使用终端命令行启动时,只会直接按照py文件文件中设置的端口号进行连接,即比如设置了端口号为8080,则只能通过8080端口号连接。我们需要动态绑定端口号,以我们需要的端口号进行连接

设置前:python3 命令行启动动态绑定端口号.py
	默认按文件中代码设置的端口号
设置后:python3 命令行启动动态绑定端口号.py 9000
	按照空格后的端口号进行绑定 即端口号为9000

实现步骤:

  1. 获取执行python程序的终端命令行参数
# 获取执行python程序的终端命令行参数
print(sys.argv) #输出为[文件名.py,端口号]
#取出输入的端口号,并转换为int型
port=sys.argv[1]
  1. 判断参数的类型,设置端口号必须是整型
# 判断格式是否错误
if len(sys.argv)!=2:
    print("格式错误,格式为:python3 文件名.py 端口号")
    return
# 判断是否是整型
if not sys.argv[1].isdigit():
    print("格式错误,端口号为整型")
    return
  1. 给Web服务器类的初始化方法添加一个端口号参数,用于绑定端口号
def __init__(self,port): #添加一个端口参数
 # 绑定端口号
	....
    self.tcp_server_socket.bind(("", port))
(2)Web静态服务器动态绑定端口号示例代码:
import socket
import threading
import sys

# 定义web服务器类
class HttpWebServer(object):
    def __init__(self,port): #添加一个端口参数
        # 创建tcp服务端套接字
        self.tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 设置端口号复用, 程序退出端口立即释放
        self.tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
        # 绑定端口号
        self.tcp_server_socket.bind(("", port))
        # 设置监听
        self.tcp_server_socket.listen(128)

    # 处理客户端的请求,放start函数前
    @staticmethod
    def handle_client_request(self,server_client_socket):
        # 获取浏览器的请求信息
        client_request_data = server_client_socket.recv(4096)
        # 对二进制数据进行解码
        client_request_data = client_request_data.decode()
        print(client_request_data)
        # 根据指定字符串进行分割
        request_list = client_request_data.split(" ")
        # 判断客户端是否关闭:
        # 若返回的数据只包含一个,则表示返回的是浏览器返回的空字符串信息告知,则表示浏览器关闭,这时直接关闭套接字就可,并直接结束函数
        if len(request_list) == 1:
            server_client_socket.close()
            return
        # 获取请求资源路径
        request_path = request_list[1]
        print(request_path)
        # 判断请求的是否是根目录,如果条件成立,指定首页数据返回
        if request_path == "/":
            request_path = "/index.html"
        try:
            # 动态打开指定文件
            with open("./Web" + request_path, "rb") as file:
                # 读取文件数据
                file_data = file.read()
        except Exception as e:
            # 请求资源不存在,返回404数据
            # 响应行
            response_line = "HTTP/1.1 404 Not Found\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            with open("./Web/error.html", "rb") as file:
                file_data = file.read()
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        else:  # 正常返回数据
            # 响应行
            response_line = "HTTP/1.1 200 OK\r\n"
            # 响应头
            response_header = "Server: PythonWeb\r\n"
            # 响应体
            response_body = file_data
            # 拼接响应报文
            response_data = (response_line + response_header + "\r\n").encode() + response_body
            # 发送数据
            server_client_socket.send(response_data)
        finally:
            # 关闭服务与客户端的套接字
            server_client_socket.close()

    # 启动Web服务器进行工作
    def start(self):
        while True:
            # 等待接受客户端的连接请求
            server_client_socket, client_addr = self.tcp_server_socket.accept()
            # 代码执行到此,说明连接建立成功
            # 创建子线程,传递参数是为了和客户端进行通信的,即传递server_client_socket,同时设置守护主线程
            sub_thread=threading.Thread(target=self.handle_client_request,args=(server_client_socket,),daemon=True)
            # 启动子线程
            sub_thread.start()

# yu
def main():
    # 获取执行python程序的终端命令行参数
    print(sys.argv) #输出为[文件名.py,端口号]
    #取出输入的端口号,并转换为int型
    port=sys.argv[1]
    # 判断格式是否错误
    if len(sys.argv)!=2:
        print("格式错误,格式为:python3 文件名.py 端口号")
        return
    # 判断是否是整型
    if not sys.argv[1].isdigit():
        print("格式错误,端口号为整型")
        return
    # 给服务器类的初始化添加一个端口号参数,用于绑定端口号
    # 创建web服务器对象
    web_server=HttpWebServer(port)
    # 启动Web服务器进行工作
    web_server.start()

if __name__=='__main__':
    main()

ept()
# 代码执行到此,说明连接建立成功
# 创建子线程,传递参数是为了和客户端进行通信的,即传递server_client_socket,同时设置守护主线程
sub_thread=threading.Thread(target=self.handle_client_request,args=(server_client_socket,),daemon=True)
# 启动子线程
sub_thread.start()

yu

def main():
# 获取执行python程序的终端命令行参数
print(sys.argv) #输出为[文件名.py,端口号]
#取出输入的端口号,并转换为int型
port=sys.argv[1]
# 判断格式是否错误
if len(sys.argv)!=2:
print(“格式错误,格式为:python3 文件名.py 端口号”)
return
# 判断是否是整型
if not sys.argv[1].isdigit():
print(“格式错误,端口号为整型”)
return
# 给服务器类的初始化添加一个端口号参数,用于绑定端口号
# 创建web服务器对象
web_server=HttpWebServer(port)
# 启动Web服务器进行工作
web_server.start()

if name==‘main’:
main()

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

PCGuo999

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值