百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

搭建Python自带静态Web服务器 python 搭建的http 动态服务器

myzbx 2024-12-17 15:13 10 浏览

静态web服务器是什么?

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

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

如何搭建Python自带的静态Web服务器

搭建Python自带的静态web服务器使用 python -m http.server 端口号,端口号不指定默认8000

-m选项说明:

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

静态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(('', 8000))
    # 设置监听
    tcp_server_socket.listen(128)
    # 循环等待接受客户端的连接请求
    while True:
        # 等待接受客户端的连接请求
        new_socket, ip_port = tcp_server_socket.accept()
        # 代码执行到此,说明连接建立成功
        # 接收客户端的请求信息
        recv_data = new_socket.recv(4096)
        print(recv_data)

        # 打开文件,读取文件中的数据
        # 这里的file表示打开文件的对象
        with open('static/index.html', 'r') as file:
            file_data = file.read()
        # 提示:with open 关闭文件这步操作不用程序员书写,系统自动完成

        # 响应行
        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' + response_body
        # 把字符串编码成二进制
        response_data = response.encode('utf-8')
        # 发送给浏览器的响应报文数据
        new_socket.send(response_data)
        # 关闭服务于客户端的套接字
        new_socket.close()

静态web服务器-返回指定页面数据

import socket
import os


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()
        # 代码执行到此,说明连接建立成功
        # 接收客户端的请求信息
        recv_data = new_socket.recv(4096)
        # 判断接收的数据长度是否为0
        if len(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)

        # 判断请求的是否是根目录,如果是,设置根目录
        if request_path == '/':
            request_path = '/index.html'

        # 1. os.path.exists
        # os.path.exists('static/' + request_path)
        # 2. try-except
        try:
            # 打开文件,读取文件中的数据,这里使用rb模式,兼容打开图片文件
            # 这里的file表示打开文件的对象
            with open('static' + request_path, 'rb') as file:
                file_data = file.read()
            # 提示:with open 关闭文件这步操作不用程序员书写,系统自动完成
        except Exception as e:
            # 代码执行到此,说明没有请求的该文件,返回404状态信息
            # 响应行
            response_line = 'HTTP/1.1 404 Not Found\r\n'
            # 响应头
            response_header = 'Server: PWS/1.0\r\n'
            # 读取404页面数据
            with open('static/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)
            pass
        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()

静态web服务器-多任务版

import socket
import threading


# 处理客户端请求
def handle_client_request(new_socket):
    # 接收客户端的请求信息
    recv_data = new_socket.recv(4096)
    # 判断接收的数据长度是否为0
    if len(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)

    # 判断请求的是否是根目录,如果是,设置根目录
    if request_path == '/':
        request_path = '/index.html'

    # 1. os.path.exists
    # os.path.exists('static/' + request_path)
    # 2. try-except
    try:
        # 打开文件,读取文件中的数据,这里使用rb模式,兼容打开图片文件
        # 这里的file表示打开文件的对象
        with open('static' + request_path, 'rb') as file:
            file_data = file.read()
        # 提示:with open 关闭文件这步操作不用程序员书写,系统自动完成
    except Exception as e:
        # 代码执行到此,说明没有请求的该文件,返回404状态信息
        # 响应行
        response_line = 'HTTP/1.1 404 Not Found\r\n'
        # 响应头
        response_header = 'Server: PWS/1.0\r\n'
        # 读取404页面数据
        with open('static/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)
        pass
    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()


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

静态web服务器-面向对象开发

实现步骤

  1. 把提供服务的web度武器抽象成一个类(HTTPWebServer)
  2. 提供web服务器的初始化方法,在初始化方法里面创建socket对象
  3. 提供一个开启web服务器的方法,让web服务器处理客户端请求操作
import socket
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

    # 启动服务器的方法
    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()

    # 处理客户端请求
    @staticmethod
    def handle_client_request(new_socket):
        # 接收客户端的请求信息
        recv_data = new_socket.recv(4096)
        # 判断接收的数据长度是否为0
        if len(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)

        # 判断请求的是否是根目录,如果是,设置根目录
        if request_path == '/':
            request_path = '/index.html'

        # 1. os.path.exists
        # os.path.exists('static/' + request_path)
        # 2. try-except
        try:
            # 打开文件,读取文件中的数据,这里使用rb模式,兼容打开图片文件
            # 这里的file表示打开文件的对象
            with open('static' + request_path, 'rb') as file:
                file_data = file.read()
            # 提示:with open 关闭文件这步操作不用程序员书写,系统自动完成
        except Exception as e:
            # 代码执行到此,说明没有请求的该文件,返回404状态信息
            # 响应行
            response_line = 'HTTP/1.1 404 Not Found\r\n'
            # 响应头
            response_header = 'Server: PWS/1.0\r\n'
            # 读取404页面数据
            with open('static/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)
            pass
        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()


def main():
    # 创建web服务器
    web_server = HttpWebServer()
    # 启动服务器
    web_server.start()


# 判断是否是主模块的代码
if __name__ == '__main__':
    main()

获取终端命令行参数

import sys


# 获取终端命令行参数
params = sys.argv

# 列表里面的每项数据都是字符串类型
print(params, type(params))

静态web服务器-命令行启动动态绑定端口号

实现步骤

  1. 获取执行python程序的终端命令行参数
  2. 判断参数的类型,设置端口号必须是整型
  3. 给web服务器的初始化方法添加一个端口号参数,用于绑定端口号
import socket
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

    # 启动服务器的方法
    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()

    # 处理客户端请求
    @staticmethod
    def handle_client_request(new_socket):
        # 接收客户端的请求信息
        recv_data = new_socket.recv(4096)
        # 判断接收的数据长度是否为0
        if len(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)

        # 判断请求的是否是根目录,如果是,设置根目录
        if request_path == '/':
            request_path = '/index.html'

        # 1. os.path.exists
        # os.path.exists('static/' + request_path)
        # 2. try-except
        try:
            # 打开文件,读取文件中的数据,这里使用rb模式,兼容打开图片文件
            # 这里的file表示打开文件的对象
            with open('static' + request_path, 'rb') as file:
                file_data = file.read()
            # 提示:with open 关闭文件这步操作不用程序员书写,系统自动完成
        except Exception as e:
            # 代码执行到此,说明没有请求的该文件,返回404状态信息
            # 响应行
            response_line = 'HTTP/1.1 404 Not Found\r\n'
            # 响应头
            response_header = 'Server: PWS/1.0\r\n'
            # 读取404页面数据
            with open('static/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)
            pass
        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()


def main():
    # 获取终端命令行参数
    params = sys.argv
    if len(params) != 2:
        print('执行的命令格式如下:python xxx.py 9000')
        return

    # 判断第二个参数是否都是由数字组成的字符串
    if not params[1].isdigit():
        print('执行的命令格式如下:python xxx.py 9000')
        return

    # 代码执行到此,说明命令行参数的个数一定是两个,并且第二个是由数字组成的字符串
    port = int(params[1])

    # 创建web服务器
    web_server = HttpWebServer(port)
    # 启动服务器
    web_server.start()


# 判断是否是主模块的代码
if __name__ == '__main__':
    main()

相关推荐

攀升战境S5电竞主机评测:NVIDIA RTX 3060实力助阵,光追游戏走起

此次笔者将为玩家们推荐一款游戏主机——攀升战境S5。该主机是攀升电脑今年力推的游戏装备,主机采用一线品牌配件,特别是在显卡选用上严苛把关,精选GeForceRTX30系列显卡,玩家们大可以放心选购...

慎买-神牛闪光灯兼容性问题:神牛V350&松下S5M2

神牛V350和松下S5M2的兼容性问题。大家好,我是向往闪光灯人像的Fish。国庆期间,我购买了神牛V350闪光灯和神牛X2T引闪器,但这成为了我的噩梦。我原以为客服和松友们说这款闪光灯在松下S5M2...

Acer蜂鸟持续办公一整天(acer 蜂鸟s5)

移动办公在工作节奏日益加快的今天越来越普遍,目前大部分工作无法在手持设备上完成,笔记本依然是移动办公最明智的选择。为了实现移动办公,很多笔记本越做越轻薄,性能也越来越强,而续航却一直没有很大提升。笔者...

职业车手明年会骑什么?2021赛季各大世巡赛车队使用器材一览

新年的钟声即将敲响,意味着充满魔幻色彩的2020年即将过去。受新冠肺炎的影响,2020年的赛季非常不同寻常。因这一原因不得不延迟举行的各种比赛导致许多车队的赞助商无法得到足够曝光,这也间接导致了许多车...

三星部分手机系统升级路线图流出(三星系统在哪升级)

三星包括Note3和S5在内的手机在升级到4.4.2系统之后一直没有什么系统升级的消息,而最近流出的一张三星的系统升级路线图中出现了一共13台手机升级KTU84P(也就是Android4.4.4)...

索尼Xperia Z3配置大曝光:升级并不大

IT之家(www.ithome.com):索尼XperiaZ3配置大曝光:升级并不大索尼明天就会在IFA2014大会上发布其下代旗舰XperiaZ3智能手机,目前网上曝光了其原型机,并且机身背后...

不进反退 三星Exynos 5433只能运行32位模式?

三星GalaxyNote4将带有两个版本,除了国行使用的骁龙805以外,还有三星自家的Exynos5433版本。而这颗SoC的详细信息三星并没有公布,据外媒Anandtech称,他们从源码中确认...

尼康Z6III测评:对比EOS R6 II、A7M4、S5IIX

摄影器材测评网站DPReview刚刚发布了尼康Z6III的完整图文测评,该机获得金奖评级,得分达到91%。以下是该文章的摘录——尼康Z6III核心规格:2400万像素“部分堆栈式”传感器RAW连拍:机...

赛默飞Ion S5首批数据公布,玩爆前任PGMTM系列

北美时间9月1日,赛默飞发布了两款最新的NGS系统IonS5和IonS5XL,旨在提供更加简捷的靶向测序流程。10月29日IonS5测序仪的首批实验数据产生于阜外医院。阜外医院研究人员选用了主...

Excel技巧:快速制作批量文件夹,省时省力,加强工作效率

大家好,如果公司领导要求按人员姓名制作文件夹,以一人一档的形式呈现人员档案,办公人员一个一个制作费时费力,而且效力低下,今天为大家介绍快捷制作批量文件夹的方法下面我们用图片来进行演示操作打开表格,选...

国行、港版、美版Apple watch各版本售价一览

今天凌晨,苹果牌手表正式发布,苹果开始正式进入可穿戴设备领域,除了功能和外观,我相信大家更关心的是价格问题了,小编就将国行、港版、美版的Applewatch售价做一总结,以供参考。国行:美版:港版:...

松下全画幅微单S5和S1到底哪里不一样?

Hello,我是ET,欢迎大家来到我的“相机笔记”。————9月2日晚,松下正式发布了第4款全画幅微单LUMIXS5。这一篇,我们主要来说松下LUMIXS5和LUMIXS1到底有哪些区别...

融会贯通之典范 神舟S7-2021S5评测

便携、性能、续航,这简简单单的六个字道出了这么些年来笔记本电脑的设计方向,可是由于底层技术、模具设计等等原因,这三点并不能很好的融合在一起。虽说闻道有先后,术业有专攻,但能够有一台融会贯通的产品,不是...

三国志战略版:S5赛季装X指南,开荒不是一成不变,需要因地制宜

大家好我是零氪玩家花席,S5赛季已经开始,因为S5赛季的野地阵容和S4赛季没有区别,所以S5赛季开荒相对不难。你在S4有经验,并且多了很多武将和战法,还能用150赛季功勋兑换7500战法点。S5赛季新...

聊聊松下S5M2和S5M2X的区别(松下s5k和s5c有什么区别)

先简单说下哪里不同:12bitRAWHDMI外录支持直接将视频录制到USB-SSD上多了All-Intra和ProRes编码支持有线/无线IP推流,USB网络连接黑化的机身不过要特别强调一下,S5...