余晖落尽暮晚霞,黄昏迟暮远山寻
本站
当前位置:网站首页 > 编程知识 > 正文

搭建Python自带静态Web服务器(python网页服务器)

xiyangw 2022-11-25 12:06 14 浏览 0 评论

静态web服务器是什么?

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

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

搭建Python自带静态Web服务器(python网页服务器)

如何搭建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()

相关推荐

spring利用spring.handlers解析自定义配置(spring validation 自定义)

一、问题我们在spring的xml配置文件里经常定义各种各样的配置(tx、bean、mvc、bean等等)。以及集成第三方框架时,也会看到一些spring之外的配置,例如dubbo的配置、securi...

「Spring源码分析」AOP源码解析(上篇)(spring源码深度解析(第2版))

前言前面写了六篇文章详细地分析了SpringBean加载流程,这部分完了之后就要进入一个比较困难的部分了,就是AOP的实现原理分析。为了探究AOP实现原理,首先定义几个类,一个Dao接口:1&nbs...

Spring 解析注册BeanDefinition这一篇就Over
Spring 解析注册BeanDefinition这一篇就Over

一、简介:学习过Spring框架的人一定都会听过Spring的IoC(控制反转)、DI(依赖注入)这两个概念,对于初学Spring的人来说,总觉得IoC、...

2023-03-20 14:53 xiyangw

域、模块、空间、闭包,你真的懂了吗?(模块控制域与作用域的关系)

Javascript有一个特性叫做域。尽管对于初学者来说理解域是有难度的,但我会尽力用最简单的方式让你理解域。理解域能让你的代码更优秀,减少错误,及有助于你做出更强大的模式设计。什么是域域是在运行时,...

这一次搞懂Spring自定义标签以及注解解析原理
这一次搞懂Spring自定义标签以及注解解析原理

前言在上一篇文章中分析了Spring是如何解析默认标签的,并封装为BeanDefinition注册到缓存中,这一篇就来看看对于像context这种自定义标签是如...

2023-03-20 14:53 xiyangw

前端基础进阶(七)-前端工程师最容易出错的问题-this关键字
前端基础进阶(七)-前端工程师最容易出错的问题-this关键字

我们在学习JavaScript的时候,因为对一些概念不是很清楚,但是又会通过一些简洁的方式把它给记下来,那么这样自己记下来的概念和真正的概念产生了很强的偏差.当...

2023-03-20 14:52 xiyangw

深入K8s:守护进程DaemonSet及其源码分析(k8s 进程)
深入K8s:守护进程DaemonSet及其源码分析(k8s 进程)

建议学习:膜拜!阿里内部都在强推的K8S(kubernetes)学习指南,不能再详细了最近也一直在加班,处理项目中的事情,发现问题越多越是感觉自己的能力不足,...

2023-03-20 14:52 xiyangw

Spring 是如何解析 bean 标签的?(spring beans标签)
Spring 是如何解析 bean 标签的?(spring beans标签)

前情回顾上回「SpringIoC容器初始化(2)」说到了Spring如何解析我们定义的<bean>标签,代码跟进了一层又一层,跋山涉水,...

2023-03-20 14:52 xiyangw

快速了解JavaScript文本框操作(javascript文本框代码)
快速了解JavaScript文本框操作(javascript文本框代码)

HTML中使用<input>元素表示单行输入框和<textarea>元素表示多行文本框。HTML中使用的<input&...

2023-03-20 14:51 xiyangw

荐读|30道JavaOOP面试题,可以和面试官扯皮了
荐读|30道JavaOOP面试题,可以和面试官扯皮了

面试是我们每个人都要经历的事情,大部分人且不止一次,今天给大家准备了30道JavaOOP面试题,希望能够帮助到对Java感兴趣的同学,让大家在找工作的时候能够...

2023-03-20 14:51 xiyangw

源码系列——mybatis源码刨析总结,下(mybatis源码分析)
源码系列——mybatis源码刨析总结,下(mybatis源码分析)

接上文简答题一.1.Mybatis动态sql是做什么的?1.动态sql就是根据条件标签动态的拼接sql,包括判空,循环,拼接等2.哪些动态sql?动态sql大...

2023-03-20 14:50 xiyangw

Java面试题(第二弹)(java面试题及答案整理)
Java面试题(第二弹)(java面试题及答案整理)

1.抽象类和接口的区别?接口可以被多重implements,抽象类只能被单一extends接口只有定义,抽象类可以有定义和实现接口的字段定义默认为:public...

2023-03-20 14:50 xiyangw

mybatis3 源码深度解析-动态 sql 实现原理(sql数据库基础知识)
mybatis3 源码深度解析-动态 sql 实现原理(sql数据库基础知识)

大纲动态sql使用示例SqlSource和BoundSql以及实现类LanguageDriver以及实现类SqlNode以及实现类动态sql解...

2023-03-20 14:50 xiyangw

第43节 Text、Comment及CDATASection(第43节 Text、Comment及CDATASection)
第43节 Text、Comment及CDATASection(第43节 Text、Comment及CDATASection)

本内容是《Web前端开发之Javascript视频》的课件,请配合大师哥《Javascript》视频课程学习。文本节点用Text类型表示,包含的是可以按字面解释...

2023-03-20 14:49 xiyangw

Qt读写三种文件(qt读取文件数据并赋值给变量)

第一种INI配置文件.ini文件是InitializationFile的缩写,即初始化文件。除了windows现在很多其他操作系统下面的应用软件也有.ini文件,用来配置应用软件以实现不同用户的要...

取消回复欢迎 发表评论: