什么是web框架
框架,即framework,特指为解决一个开放性问题而设计的具有一定约束性的支撑结构,使用框架可以帮你快速开发特定的系统,简单地说,就是你用别人搭建好的舞台来做表演。
对于所有的Web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端。
最简单的web框架
import socket def handle_request(client): buf = client.recv(1024) client.send("HTTP/1.1 200 OK\r\n\r\n".encode("utf8")) client.send("<h1 style='color:red'>Hello, Python</h1>".encode("utf8")) def main(): sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.bind(('localhost',8001)) sock.listen(5) while True: connection, address = sock.accept() handle_request(connection) connection.close() if __name__ == '__main__': main()
最简单的Web应用就是先把HTML用文件保存好,用一个现成的HTTP服务器软件,接收用户请求,从文件中读取HTML,返回。
但是这些代码太过繁琐,因为我们不希望接触到TCP连接、HTTP原始请求和响应格式,所以,需要一个统一的接口,让我们专心用Python编写Web业务。
这个接口就是WSGI(web网关接口协议):Web Server Gateway Interface。
利用WSGI搭建web框架
from wsgiref.simple_server import make_server def application(environ, start_response): start_response('200 OK', [('Content-Type', 'text/html')]) return [b'<h1>Hello, web!</h1>'] httpd = make_server('', 8080, application) print('Serving HTTP on port 8000...') # 开始监听HTTP请求: httpd.serve_forever()
在平时的使用中,我们会在浏览器输入地址中加入路径相关信息,所以这里对原有框架进行升级。
在之前存储着请求信息大字典的environ变量中,有一个"PATH_INFO"的键,可以获取路径相关信息。下面是关于原始服务器的升级:
from wsgiref.simple_server import make_server # 使用这个库之后,不需要对传过来的请求进行解析,它自动帮我们进行处理 def application(environ, start_response): # 一旦有客户端连接时,自动执行 # 浏览器传过来的信息会被处理为一个大字典放入到environ中。 start_response('200 OK', [('Content-Type', 'text/html')]) # 设置相应的请求头 # f1=open("index1.html","rb") # data1=f1.read() # f2=open("index2.html","rb") # data2=f2.read() path = environ['PATH_INFO'] if path == "/111.html": return [b'<h1>Hello, web</h1>'] # return [data1] elif path == "/222.html": return [b'<h1>Hello, Django</h1>'] # return [data2] else: return ["<h1>404</h1>".encode('utf8')] # return [b'<h1>Hello, web!</h1>'] # 返回响应结果 httpd = make_server('', 8080, application) # 这里需要传入三个参数:ip地址,端口号,应用 print('Serving HTTP on port 8000...') # 开始监听HTTP请求: httpd.serve_forever()
下一步
from wsgiref.simple_server import make_server # 使用这个库之后,不需要对传过来的请求进行解析,它自动帮我们进行处理 def f1(): f1=open("index1.html","rb") data1=f1.read() return [data1] def f2(): f2=open("index2.html","rb") data2=f2.read() return [data2] def application(environ, start_response): # 一旦有客户端连接时,自动执行 # 浏览器传过来的信息会被处理为一个大字典放入到environ中。 start_response('200 OK', [('Content-Type', 'text/html')]) # 设置相应的请求头 # f1=open("index1.html","rb") # data1=f1.read() # f2=open("index2.html","rb") # data2=f2.read() path = environ['PATH_INFO'] if path == "/111.html": return f1() # return [b'<h1>Hello, web</h1>'] # return [data1] elif path == "/222.html": return f2() # return [b'<h1>Hello, Django</h1>'] # return [data2] else: return ["<h1>404</h1>".encode("utf-8")] # return [b'<h1>Hello, web!</h1>'] # 返回响应结果 httpd = make_server('', 8080, application) # 这里需要传入三个参数:ip地址,端口号,应用 print('Serving HTTP on port 8000...') # 开始监听HTTP请求: httpd.serve_forever()
路由分发
from wsgiref.simple_server import make_server # 使用这个库之后,不需要对传过来的请求进行解析,它自动帮我们进行处理 def r1(): f1 = open("111.html", "rb") data1 = f1.read() return [data1] def r2(): f2 = open("222.html", "rb") data2 = f2.read() return [data2] # 完成路由分发 def router(): url_patterns = [ ("/111.html", r1), ("/222.html", r2), ] return url_patterns def application(environ, start_response): # 一旦有客户端连接时,自动执行 # 浏览器传过来的信息会被处理为一个大字典放入到environ中。 start_response('200 OK', [('Content-Type', 'text/html')]) # 设置相应的请求头 path = environ['PATH_INFO'] # 获取URL中的路径 url_patterns = router() func = None for i in url_patterns: if i[0] == path: func = i[1] return func() if func == None: return ['<h1> 404 </h1>'.encode("utf-8")] httpd = make_server('', 8080, application) # 这里需要传入三个参数:ip地址,端口号,应用 print('Serving HTTP on port 8000...') # 开始监听HTTP请求: httpd.serve_forever()
到现在,整个web框架就算完成了,接下来加入实时时间的功能,显示在前端页面
from wsgiref.simple_server import make_server # 使用这个库之后,不需要对传过来的请求进行解析,它自动帮我们进行处理 import time def f1(req): f1=open("index1.html","rb") data1=f1.read() return [data1] def f2(req): print(req["QUERY_STRING"]) # 获取表单中传来的信息 f2=open("index2.html","rb") data2=f2.read() return [data2] def show_time(req): times = time.ctime() # return [b"<h1>time:%s</h1>" % times] # 小问题,这种写法是否正确? # return [("<h1>time:%s</h1>" % str(times)).encode("utf-8")] # 下面是纯英文内容,如果有中文会出现乱码 # f = open("show_time.html", "r") # data = f.read() # data = data.replace("{{time}}", times) # return [data.encode("utf-8")] # 乱码解决 f = open("show_time.html", "rb") data = f.read() data = data.decode("utf-8") data = data.replace("{{time}}", times) return [data.encode("utf-8")] def router(): url_patterns = [ ("/111.html", f1), ("/222.html", f2), ("/show_time", show_time) ] return url_patterns def application(environ, start_response): # 一旦有客户端连接时,自动执行 # 浏览器传过来的信息会被处理为一个大字典放入到environ中。 start_response('200 OK', [('Content-Type', 'text/html')]) # 设置相应的请求头 path = environ['PATH_INFO'] url_patterns = router() for i in url_patterns: if i[0] == path: func = i[1] return func(environ) else: return ['<h1> 404 </h1>'.encode("utf-8")] httpd = make_server('', 8080, application) # 这里需要传入三个参数:ip地址,端口号,应用 print('Serving HTTP on port 8000...') # 开始监听HTTP请求: httpd.serve_forever()