python第七课

WSGI

Middleware

中间件,可以理解称对应用程序的一组装饰器,对两边都起作用的元素。

  • 重写environ,然后基于URL,将请求对象路由给不同的应用对象
  • 支持多个应用或者框架顺序地运行于同一个进程中
  • 通过转发请求和相应,支持负载均衡和远程处理
  • 支持对内容做后处理

应用程序看来,它可以提供一个类start_response函数,可以像start_response函数一样接收HTTP STATU和Headers和environ。

服务端看来,它可以接收2个参数,并且可以返回一个类Application对象。

练习:实现记录每次请求消耗时间功能的中间件

import time # time模块,获取当前时间,以便计算请求的处理时间
# 来自wsgiref.simple_server模块的make_server函数,用于创建一个简单的WSGI服务器
from wsgiref.simple_server import make_serverHELLO_WORLD = b'Hello, World!\n'# doucment.charset 网页摁F12在控制台输出该指令查询编码格式class ResponseTimingMiddleware(object): # 中间件类def __init__(self, app):
        self.app = appdef __call__(self, env, start_response): # __call__方法使得类的实例可以像函数一样被调用
        start_time = time.time()
        response = self.app(env, start_response)
        end_time = time.time()
        res_time = (end_time - start_time) * 1000 # 将秒转换为毫秒# 保留十位小数
        timing_text = "记录请求耗时中间件输出\n\n本次请求耗时:{:.10f}ms \n\n".format(res_time)
        # 将生成的时间记录信息转换为字节类型(使用encode()方法),并将其添加到相应内容response中
        response.append(timing_text.encode())return responsedef simple_app(environ, start_response):
    # 状态码
    status = '200 OK'
    # 设置Content-type响应头,明确字符编码为 utf-8(解决中文乱码问题) 
    response_header = [('Content-type', 'text/plain; charset=utf-8')] # 格式固定    return_body = []# environ 是一个包含了所有HTTP请求信息的字典for k, v in environ.items():
        return_body.append("{}: {}".format(k, v))    start_response(status, response_header)return ["\n".join(return_body).encode()]# 创建应用程序
app = ResponseTimingMiddleware(simple_app)server=make_server('127.0.0.1',8080,app=app)  # 127.0.0.1  app(env, start_response)
server.serve_forever()

  • 中间件类 ResponseTimingMiddleware:
  • __init__ 方法:初始化中间件类,接收一个 WSGI 应用程序 app 作为参数,并将其存储在实例属性 self.app 中。
  • __call__ 方法:使得类的实例可以像函数一样被调用。
  • 该方法在每次处理请求时被调用,具体步骤如下:
    • start_time = time.time()记录请求开始的时间。
    • response = self.app(env, start_response)调用原始的 WSGI 应用程序处理请求,并获取响应内容。
    • end_time = time.time()记录请求结束的时间。
    • res_time = (end_time - start_time) * 1000计算请求的处理时间,并将其转换为毫秒。
    • timing_text生成包含请求处理时间的文本信息,保留十位小数。
    • response.append(timing_text.encode())将时间记录信息转换为字节类型,并添加到响应内容中。
    • return response返回包含时间记录信息的响应内容。
  • WSGI 应用程序 simple_app:

  • environ是一个包含了所有 HTTP 请求信息字典,如请求方法、请求 URL、请求头信息等。
  • start_response是一个回调函数,用于设置响应的状态码和响应头
  • status设置响应的状态码为“ 200 OK”,表示请求成功。
  • response_header设置响应头,指定响应内容的类型为纯文本,并使用 UTF-8 编码,以解决中文乱码问题。
  • return_body遍历 environ 字典,将每个键值对转换为字符串,并添加到 return_body 列表中。
  • start_response(status, response_header)调用 start_response 回调函数,设置响应的状态码和响应头。
  • return ["\n".join(return_body).encode()]将 return_body 列表中的字符串用换行符连接起来,转换为字节类型,并作为响应内容返回。
  • 创建应用程序并启动服务器:
  • app = ResponseTimingMiddleware(simple_app)将 simple_app 应用程序包装在 ResponseTimingMiddleware 中间件中,创建一个新的应用程序
  • make_server('127.0.0.1', 8080, app=app)创建一个简单的 WSGI 服务器,监听本地地址 127.0.0.1 的 8080 端口,并使用 app 作为处理请求的应用程序
  • server.serve_forever()启动服务器,并使其一直运行,不断处理接收到的请求。

运行这段 Python 代码后,会启动一个简单的 WSGI(Web Server Gateway Interface)服务器,该服务器监听本地的 8080 端口,处理客户端的 HTTP 请求,并在响应中添加请求处理时间的记录信息

Webpy

web.py是一款轻量级的python web开发框架,简单、高效、学习成本低,特别适合作为python web开发的入门框架

app.py

import web
#每个web.py应用必须先导入import web模块#URL处理
#告诉webpy怎么组织URL
#第一部分是匹配URL的正则表达式
#第二部分是接受请求的类名称
#
urls 是一个元组,用于定义 Web 应用的路由规则。路由规则决定了不同的 URL 请求将由哪个类或函数来处理'/(.*)':这是一个正则表达式/表示根路径(.*)表示捕获根路径之后任意字符序列(包括空字符)。该正则表达式的作用匹配所有的 URL 请求'Hello':表示当匹配到上述正则表达式的 URL 请求时,将由名为 Hello的类处理该请求urls = (    '/(.*)', 'hello'
)#创建一个列举这些URL的application
# globals():这是 Python 的内置函数,返回当前全局符号表的字典。
web.py会使用这个字典来查找路由规则中指定的处理类(如 Hello 类)app = web.application(urls, globals())class hello:        # 在 web.py中,GET方法用于处理 HTTP GET 请求
    def GET(self, name): # name是从 URL 中捕获的参数(由正则表达式 (.*)捕获)if not name: 
            name = 'World'return 'Hello, ' + name + '!'application = app.wsgifunc()
#
app.wsgifunc():这是 web.py提供的一个方法,用于将 web.py应用实例转换为符合 WSGI 协议的可调用对象application:将转换后的 WSGI 可调用对象赋值给 application 变量。这样做的目的是为了让应用可以在支持 WSGI 的服务器(如 Gunicorn、uWSGI 等)上部署和运行。# 让web.py启动网页应用if __name__ == "__main__":
    app.run() 
#
调用 web.py应用实例的 run()方法,启动一个内置的简单 Web 服务器,监听默认端口(通常是 8080),开始处理客户端的 HTTP 请求


 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/69872.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

RAII(Resource Acquisition Is Initialization)机制

RAII(Resource Acquisition Is Initialization)机制 1. 什么是 RAII? 🌟 RAII(资源获取即初始化,Resource Acquisition Is Initialization) 是 C 语言中的一种管理资源的编程技巧。 RAII 使资…

【kafka系列】日志存储设计 消息写入、读取

目录 日志存储设计 1. 日志存储的目录结构 2. 日志内容格式设计 3. 日志索引设计 4. 设计优势 消息写入流程 示例 流程图 消息读取流程 示例 关键设计细节 流程图 日志存储设计 Kafka的日志存储是其高吞吐、持久化能力的核心设计,其结构包含目录组织、…

vue3.x 自定义hook函数详细解读

1. 什么是自定义 Hook 函数? 自定义 Hook 函数是一个封装了逻辑的 JavaScript 函数,它可以使用 Vue 3 的 Composition API 提供的响应式数据和生命周期钩子。通过自定义 Hook,你可以将组件的逻辑拆分成更小、更可复用的单元。 特点&#xf…

是时候说再见了

说再见 2018 to 2025 2018:学习 2018年开始读研。师兄师姐们说可以写写CSDN博客,对找工作也有帮助。于是在12月4日,发布了自己的第一篇文章[翻译] 神经网络与深度学习 首页 - Index。当时还在学习各种基础知识,看到了这个英文文…

蓝桥杯篇---IAP15F2K61S2定时器

文章目录 前言简介定时器的工作模式1.模式02.模式13.模式24.模式3 定时器的寄存器1.TMOD2.TCON3.THO/TL04.TH1/TL1 定时器的使用步骤1.配置TMOD2.设置初值3.启动定时器4.使能中断5.编写中断服务函数 示例代码:定时器的基本使用代码说明示例代码:定时器1用…

2D 游戏艺术、动画和光照

原文:https://unity.com/resources/2d-game-art-animation-lighting-for-artists-ebook 笔记 用Tilemap瓷砖大小为1单元,人物大小在0.5~2单元 PPU :单位像素 pixels per unit 2160 4K分辨率/ 正交相机size*2 完整屏幕显示像素点 有骨骼动…

HTML的入门

一、HTML HTML(HyperText Markup Language,超文本标记语言)是一种用来告知浏览器如何组织页面的标记语言。 超文本:就是超越了文本;HTML不仅仅可以用来显示文本(字符串、数字之类),还可以显示视频、音频等…

C语言:指针详解

C语言:指针详解 1:指针的基本概念1:什么是指针2:为什么要引入指针3:指针的作用4:指针的类型 2:指针的声明与初始化1: 指针的声明2: 指针的初始化 3:指针的操作…

Spring Boot “约定大于配置”

什么是“约定大于配置”? “约定大于配置”是一种简化开发的设计理念。简单来说,就是框架默认提供了常见的配置和行为,开发者只需要按照约定来编写代码,避免了繁琐的配置,只在需要时进行定制和调整。这种理念在Spring…

redis sentinel模式 与 redis 分片集群 配置

Redis 最低为5.0版本,以下为6.2.6版本信息。 模式 高可用性 数据分片 部署复杂度 适用场景 Sentinel 模式 高 无 中等 中小规模,需要高可用性 集群模式 高 支持 复杂 大规模,需要高…

HCIA项目实践---OSPF的基本配置

9.5.12 OSPF的基本配置 (所搭环境如上图所示) A 先配置IP地址 (先进入路由器R1的0/0/0接口配置IP地址,再进入环回接口配置IP地址) (配置R2路由器的0/0/0和0/0/1以及环回接口的IP地址) (置R3路由器的0/0/0接…

【MyBatis】预编译SQL与即时SQL

目录 1. 以基本类型参数为例测试#{ }与${ }传递参数的区别 1.1 参数为Integer类型 1.2 参数为String类型 2. 使用#{ }传参存在的问题 2.1 参数为排序方式 2.2 模糊查询 3. 使用${ }传参存在的问题 3.1 SQL注入 3.2 对比#{ } 与 ${ }在SQL注入方面存在的问题 3.3 预编译…

07:串口通信(二):收发数据包

1、数据包 我们使用上位机个单片机发送数据包时,规定包头和包尾,将我们需要发送的数据放在中间,数据的长度我们也可以自己规定。一般情况下HEX数据包我们使用固定长度数据包。而文本数据包使用是可变长度数据包。 2、HEX数据包 2.1、HEX固定…

vs2022支持.netframework4.0

下载nuget包 .netframework4.0 解压nuget 复制到C:\Program Files (x86)\Reference Assemblies\Microsoft\Framework\.NETFramework 参考 https://www.cnblogs.com/bdqczhl/p/18670152 https://blog.csdn.net/xiaomeng1998_/article/details/135979884

《安富莱嵌入式周报》第350期:Google开源Pebble智能手表,开源模块化机器人平台,开源万用表,支持10GHz HRTIM的单片机,开源CNC控制器

周报汇总地址:嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 视频版: https://www.bilibili.com/video/BV1YPKEeyEeM/ 《安富莱嵌入式周报》第350期:Google开…

Oracle临时表空间(基础操作)

临时表空间 临时表空间:用来存放用户的临时数据,临时数据在需要时被覆盖,关闭数据库后自动删除,其中不能存放永久性数据。 用户进程和服务器进程是一对一的叫做专用连接。 任何一个用户连到oracle数据库,oracle都会…

浅识MQ的 Kafka、ActiveMQ、RabbitMQ、RocketMQ区别

DeepSeek回复: 以下是主流消息队列(MQ)的对比分析,结合核心特性、适用场景和实际案例说明: 一、主流MQ对比分析 维度 Kafka RabbitMQ RocketMQ ActiveMQ所属公司Apache(LinkedIn开源)…

19.4.2 -19.4.4 新增、修改、删除数据

版权声明:本文为博主原创文章,转载请在显著位置标明本文出处以及作者网名,未经作者允许不得用于商业目的。 需要北风数据库的请留言自己的信箱。 19.4.2 新增数据 数据库数据的新增、修改和删除不同于查询,查询需要返回一个DbD…

若依系统环境搭建记录

开源若依系统网上资料也很全的,本篇博文记录下自己搭建环境过程中遇到的一些问题。 配置Maven和编辑器选择 我懒得配置Eclipse了,直接用vscode作为编辑器,后面构建运行都用命令行。 配置数据库连接 按照mysql5.7按网上教程即可&#xff1…

Redis——优惠券秒杀问题(分布式id、一人多单超卖、乐悲锁、CAS、分布式锁、Redisson)

#想cry 好想cry 目录 1 全局唯一id 1.1 自增ID存在的问题 1.2 分布式ID的需求 1.3 分布式ID的实现方式 1.4 自定义分布式ID生成器(示例) 1.5 总结 2 优惠券秒杀接口实现 3 单体系统下一人多单超卖问题及解决方案 3.1 问题背景 3.2 超卖问题的…