使用Web框架
了解了WSGI框架,我们发现,其实一个Web APP,就是写一个WSGI处理函数,针对每一个HTTP请求响应。
但是如何处理HTTP请求不是问题,问题是如何同时处理100个不同URL。
每一个URL可以对应GET和POST请求,当然还有DELETE、PUT等请求,但是我们通常只考虑GET和POST请求。
一个最简单的想法是从environ中取出HTTP请求,然后逐个判断:
def application(environ,start_response):
method =environ('REQUEST_METHOD')
path =environ('PATH_INFO')
if method =='GET' and path =='/':
return handle_home(environ,start_response)
if method =='POST' and path =='/signin':
return handle_signin(environ,start_response)
只是这样写下去,代码就无法维护了。
原因是WSGI提供的接口虽然比HTTP高级了很多,但和Web APP处理的逻辑相比,还是比较低级的,我们需要在WSGI之上进一步抽象,让我们专注用一个函数处理一个URL,至于URL到函数的映射,就交给Web框架来做。
由于用Python开发一个Web框架十分容易,所以Python有上百个开源的Web框架。我们先选择一个比较流行的Web框架-Flask来使用。
用Flask编写Web APP比使用WSGI简单,我们先用pip安装Flask:
pip install Flask
然后写一个app.py,处理3个URL,分别是:
GET /:首页,返回Home;
GET /signin :登陆页,显示登陆表单;
POST /signin:处理登陆表单,显示登陆结果。
注意,同一个URL/signin分别有GET和POST两种请求,映射到两个处理函数中。
Flask通过Python的装饰器在内部自动的把URL和函数关联起来,所以,我们写出来的代码就像这样:
from flask import Flask
from flask import request
app =Flask(__name__)
@app.route('/',methods =['GET','POST'])
def home():
return <h1>home</h1>
@app.route('/signin',methods =['GET'])
return '''<form action="/signin" method="post">
<p><input name="username"></p>
<p><input name="password" type="password"></p>
<p><button type="submit">Sign In</button></p>
</form>'''
@app.route('/signin', methods=['POST'])
def signin():
# 需要从request对象读取表单内容:
if request.form['username']=='admin' and request.form['password']=='password':
return '<h3>Hello, admin!</h3>'
return '<h3>Bad username or password.</h3>'
if __name__ == '__main__':
app.run()
运行app.py
python app.py
打开浏览器,输入:http://localhost:5000/
首页显示正确。
再次输入:http://localhost:5000/signin/
输入预设的admin和密码password
登陆成功!
输入其他错误的用户名或密码:
实际的Web APP 应该拿到用户名和口令后,去数据库查询再对比,来判断用户是否能登陆。
除了Flask之外,常见的Python Web框架还有:
Django:全能型Web框架;
web.py:一个小巧的web框架;
Bottle:和Flask类似的框架;
Tornado:FaceBook的开源异步框架。
当然,开发Python Web框架也不是什么难事,我们后面也会降到Web框架开发内容。
小结:
有了Web框架,我们在编写Web应用时,注意力就从WSGI处理函数转移到URL+对应的处理函数u,这样,编写Web APP就更简单了。
在编写URL处理函数时,除了配置URL外,从HTTP请求拿到用户数据也是非常重要的。Web框架都提供了自己的API来实现该功能。Flask通过request.form['name']来获取表单内容。