python基础-第十三篇-13.1web框架本质
Posted 财经知识狂魔
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python基础-第十三篇-13.1web框架本质相关的知识,希望对你有一定的参考价值。
基础与概念
众所周知,对于所有的web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端
web框架分两类:一类是包括socket和业务逻辑(tornado),另一类就是只负责业务逻辑
对于第二类,没有socket就要使用其他的服务器程序,比如wsgi,它负责封装客户的请求信息,我们只要按它提供的方法获取数据
首先我们明白了其本质就是socket,如果从socket开始开发应用程序那么效率就太低了,正确的做法是底层的socket处理代码由专门的服务器软件实现,而对于真实开发中的python web程序来说也是一般会分为两个部分:服务器程序和应用程序,服务器程序负责对socket服务器进行封装,并在请求到来时,对请求的各种数据进行整理,应用程序则负责具体的逻辑处理。
WSGI(Web Server Gateway Interface)是一种规范,它定义了使用python编写的web app与web server之间接口格式,实现web app与web server间的解耦。
import socket def handle_request(client): #接收客户端信息并进行处理 buf = client.recv(1024) client.send("HTTP/1.1 200 OK\\r\\n\\r\\n".encode()) client.send("Hello, Seven".encode()) def main(): # 实例socket对象 sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 绑定端口 sock.bind((\'localhost\',8888)) # 设置监听客户端数量 sock.listen(5) while True: # 获取客户端socket对象和端口 connection, address = sock.accept() handle_request(connection) connection.close() if __name__ == \'__main__\': main()
python标准库提供的独立WSGI服务器称为wsgiref
from wsgiref.simple_server import make_server def RunServer(environ, start_response): start_response(\'200 OK\', [(\'Content-Type\', \'text/html\')]) return [\'<h1>Hello, web!</h1>\'.encode(\'utf-8\')] #py3 # return \'<h1>Hello, web!</h1>\' py2 if __name__ == \'__main__\': httpd = make_server(\'\', 8888, RunServer) print("Serving HTTP on port 8888...") httpd.serve_forever()
自定义Web框架
一、框架
通过python标准库提供的wsgiref模块开发一个自己的web框架
from wsgiref.simple_server import make_server def index(): return [\'index\'.encode("utf-8")] def login(): return [\'login\'.encode("utf-8")] def routers(): # 路由函数 urlpatterns = ( (\'/index/\',index), (\'/login/\',login), ) return urlpatterns def RunServer(environ, start_response): # 响应头 start_response(\'200 OK\', [(\'Content-Type\', \'text/html\')]) # 获取客户端url信息 url = environ[\'PATH_INFO\'] # 执行路由函数,获取路由元组 urlpatterns = routers() func = None for item in urlpatterns: # 判断当前url,对应获取函数名 if item[0] == url: func = item[1] break if func: return func() else: return [\'404 not found\'.encode("utf-8")] if __name__ == \'__main__\': httpd = make_server(\'\', 8888, RunServer) print("Serving HTTP on port 8888...") httpd.serve_forever()
2、模板引擎
在上一步骤中,返回到浏览器给客户看的只是一个简单的字符串,而在真实的生活场景里,我们看到更多的是一个复杂的HTML规则字符串,所以我们一般将要返回给客户的HTML写在指定文件中,然后返回
from wsgiref.simple_server import make_server def index(): # return \'index\' f = open(\'views/index.html\') data = [f.read().encode()] return data def login(): # return \'login\' f = open(\'views/login.html\') data = [f.read().encode()] return data def routers(): urlpatterns = ( (\'/index/\', index), (\'/login/\', login), ) return urlpatterns def run_server(environ, start_response): start_response(\'200 OK\', [(\'Content-Type\', \'text/html\')]) url = environ[\'PATH_INFO\'] urlpatterns = routers() func = None for item in urlpatterns: if item[0] == url: func = item[1] break if func: return func() else: return [\'404 not found\'.encode()] if __name__ == \'__main__\': httpd = make_server(\'\', 8888, run_server) print("Serving HTTP on port 8888...") httpd.serve_forever()
可能说,你对这还不满意,因为只是一个静态页面,根本就没有动态效果,好,那怎么给客户返回动态内容??
- 自定义一套特殊的语法,进行替换
- 使用开源工具jinja2,遵循其指定语法
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> </head> <body> <h1>{{name}}</h1> {% for item in user_list %} <li>{{item}}</li> {% endfor %} </body> </html>
from wsgiref.simple_server import make_server from jinja2 import Template def index(): # return \'index\' # template = Template(\'Hello {{ name }}!\') # result = template.render(name=\'John Doe\') f = open(\'views\\index.html\') result = f.read() template = Template(result) data = template.render(name=\'John Doe\', user_list=[\'alex\', \'eric\']) return [data.encode(\'utf-8\')] def login(): # return \'login\' f = open(\'views/login.html\') data = [f.read().encode("utf-8")] return data def routers(): urlpatterns = ( (\'/index/\', index), (\'/login/\', login), ) return urlpatterns def run_server(environ, start_response): start_response(\'200 OK\', [(\'Content-Type\', \'text/html\')]) url = environ[\'PATH_INFO\'] urlpatterns = routers() func = None for item in urlpatterns: if item[0] == url: func = item[1] break if func: return func() else: return [\'404 not found\'.encode("utf-8")] if __name__ == \'__main__\': httpd = make_server(\'\', 8888, run_server) print("Serving HTTP on port 8888...") httpd.serve_forever()
一个web框架应该包括路由系统和模板引擎等基本的东西,让我们看看一个高仿真自定义web框架:
from wsgiref.simple_server import make_server import time def new(): f = open(\'s1.html\', \'r\') data = f.read() f.close() # 模拟模板引擎处理数据,将html里文件的item替换成其他数据 # 这里仅仅是用字符串的替换来最简单的说明下原理 new_data = data.replace("{{item}}", str(time.time())) return new_data def index(): f = open(\'index.html\', \'r\') data = f.read() f.close() return data def home(): return \'home\' URLS = { # 定义一个字典,简单模拟路由系统 "/new": new, # 一个url对应一个函数处理 "/index": index, "/home": home, } def application(environ, start_response): start_response(\'200 OK\', [(\'Content-Type\', \'text/html\')]) url = environ[\'PATH_INFO\'] if url in URLS.keys(): # 不同的url执行不同的函数 func_name = URLS[url] ret = func_name() else: ret = "404" return [ret.encode(\'utf-8\')] # 最后将结果返回 httpd = make_server(\'\', 8000, application) httpd.serve_forever()
以上是关于python基础-第十三篇-13.1web框架本质的主要内容,如果未能解决你的问题,请参考以下文章