09 基于模块wsgiref版web框架
09 基于模块wsgiref版web框架
模块引入
真实开发中的python web程序,一般会分为两部分:
服务器程序:负责对socket服务器进行封装,并在请求到来时,对请求的各种数据进行整理;
应用程序:负责具体的逻辑处理。
为了方便应用程序的开发,就出现了众多的Web框架,例如:Django、Flask、web.py 等。不同的框架有不同的开发方式,但是无论如何,开发出的应用程序都要和服务器程序配合,才能为用户提供服务。
这样,服务器程序就需要为不同的框架提供不同的支持。这样混乱的局面无论对于服务器还是框架,都是不好的。对服务器来说,需要支持各种不同框架,对框架来说,只有支持它的服务器才能被开发出的应用使用。最简单的Web应用就是先把HTML用文件保存好,用一个现成的HTTP服务器软件,接收用户请求,从文件中读取HTML,返回。如果要动态生成HTML,就需要把上述步骤自己来实现。不过,接受HTTP请求、解析HTTP请求、发送HTTP响应都是苦力活,如果我们自己来写这些底层代码,还没开始写动态HTML呢,就得花个把月去读HTTP规范。
正确的做法是底层代码由专门的服务器软件实现,我们用Python专注于生成HTML文档。因为我们不希望接触到TCP连接、HTTP原始请求和响应格式,所以,需要一个统一的接口协议来实现这样的服务器软件,让我们专心用Python编写Web业务。
这时候,标准化就变得尤为重要。我们可以设立一个标准,只要服务器程序支持这个标准,框架也支持这个标准,那么他们就可以配合使用。一旦标准确定,双方各自实现。这样,服务器可以支持更多支持标准的框架,框架也可以使用更多支持标准的服务器。
WSGI(Web Server Gateway Interface)就是一种规范,它定义了使用Python编写的web应用程序与web服务器程序之间的接口格式,实现web应用程序与web服务器程序间的解耦。
常用的WSGI服务器有uwsgi、Gunicorn。而Python标准库提供的独立WSGI服务器叫wsgiref,Django开发环境用的就是这个模块来做服务器。
服务器server端python程序(wsgiref模版):
from wsgiref.simple_server import make_server def application(environ,start_response): """ wsgiref.simple_server模块中的make_server方法调用的函数 :param environ: wsgiref模块中封装好的包含http请求所有信息的字典 :param start_response: wsgiref提供的内置方法(用于响应格式的发送) :return:返回给浏览器的信息(以[]包装的bytes类型) """ # environ变量是模块自行封装好的http请求信息,以键值对的形式存在,可以直接字典取值 print(environ) print(environ["PATH_INFO"]) # 打印请求的文件路径 print(environ["REQUEST_METHOD"]) # 打印请求的方式 # 服务端响应格式信息(状态信息和请求头部) start_response("200 OK",[("Content-Type","text/html")]) return [b'<h1>hello</h1>'] # 服务端返回数据用[]包装起来 #进行bind端口绑定、listen监听、封装http请求信息、 # 调用application函数(自动传递封装好的请求信息字典和wsgiref模块内置的响应格式发送函数) htttpd=make_server("127.0.0.1",8888,application) htttpd.serve_forever() #accept获取客户端端的连接,开启线程、发送服务端的返回信息
服务器server端python程序(wsgiref模版)
wsgiref模块:
将http请求封装成以键值对的形式封装成字典environ:
"PATH_INFO"对应的值为请求文件路径,
“QUERY_STRING”对应的值为get请求发送的数据
“REQUEST+METHOD”对应的值为请求方式。
同时封装一个发送响应格式的函数start_server;
通过特定的方式return [ ]返回数据并调用内置方法返回给浏览器。
客户端浏览器访问:
在浏览器地址栏直接输入地址端口:127.0.0.1:8888
基于wsgiref模块返回网页实例:
from wsgiref.simple_server import make_server def func_html(): with open("index.html","rb")as f: return f.read() def func_ico(): with open("favicon.ico","rb")as f: return f.read() # make_sever自动调用的方法 def application(environ,start_response): # 从封装好的请求信息字典找到请求文件路径 path = environ["PATH_INFO"] # 发送响应信息(状态、响应头部信息可有可无) start_response("200 OK",[("Content-Type","text/html")]) # data表示要返回给浏览器的数据信息 for mapper in func_mappers: if path == mapper[0]: data=mapper[1]() break else: data=b"Sorry:404 not find" # 固定格式返回 return [data] # 请求文件路径对应的返回数据读取操作映射 func_mappers=[ ("/",func_html), ("/favicon.ico",func_ico) ] # 程序启动位置: httpd=make_server("127.0.0.1",8888,application) httpd.serve_forever()
基于wsgiref模块返回网页实例
实例相关文件:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <link rel="icon" href="favicon.ico"> <style> span{ background-color: yellow; color: red; font-size: 20px } </style> <title>动态页面</title> </head> <body> <div> <h1>本次访问时间为: <span>#time#</span> </h1> </div> </body> </html>
index.html
最新文章
- iOS之按钮出现时加一个动画效果
- Cesium原理篇:6 Render模块(3: Shader)
- eclipse js 引用跳转
- dedecms 蜘蛛抓取设置 robots.txt
- 通过WebBrowser取得AJAX后的网页
- AS3.0的int uint Number的使用原则
- Bootstrap_表单_图像
- Zend Studio 如何配置本地apache服务器使用xdebug调试php脚本
- Datamatrix码
- 小猪猪逆袭成博士之C++基础篇(二) 常量、处理类型、自定义头文件
- servlet与Javabean之间的区别
- Lucene全文检索引擎
- Codeforces Beta Round #1 A,B,C
- Java工程师修炼之路(校招总结)
- JAVA 导包,使用前面的类~
- boost asio 学习(五) 错误处理
- MT【71】数列裂项放缩题
- MySQL 5.6 Index Condition Pushdown
- 内核加载模块时提示usb_common: exports duplicate symbol of_usb_get_dr_mode
- sublime 下面开发