这是我参与8月更文挑战的第23天,活动详情查看:8月更文挑战
URL 详解
URL是Uniform Resource Locator的简写,统一资源定位符。
一个URL由以下几部分组成:
- scheme://host:port/path/?query-string=xxx#anchor
- scheme:代表的是访问的协议,一般为http或者https以及ftp等。
- host:主机名,域名,比如www.baidu.com。
- port:端口号。当你访问一个网站的时候,浏览器默认使用80端口。
- path:查找路径。比如:www.jianshu.com/trending/no…
- query-string:查询字符串,比如:www.baidu.com/s?wd=python…
- anchor:锚点,后台一般不用管,前端用来做页面定位的。 注意:URL中的所有字符都是ASCII字符集,如果出现非ASCII字符,比如中文,浏览器会进行编码再进行传输。
web服务器和应用服务器以及web应用框架
- web服务器:负责处理http请求,响应静态文件,常见的有Apache,Nginx以及微软的IIS.
- 应用服务器:负责处理逻辑的服务器。比如php、python的代码,是不能直接通过nginx这种web服务器来处理的,只能通过应用服务器来处理,常见的应用服务器有uwsgi、tomcat等。
- web应用框架:一般使用某种语言,封装了常用的web功能的框架就是web应用框架,flask、Django以及Java中的SSH框架都是web应用框架。
Content-type和Mime-type的作用和区别
两者都是指定服务器和客户端之间传输数据的类型,区别如下:
- Content-type:既可以指定传输数据的类型,也可以指定数据的编码类型,例如:text/html;charset=utf-8
- Mime-type:不能指定传输的数据编码类型。例如:text/html
常用的数据类型如下:
- text/html(默认的,html文件)
- text/plain(纯文本)
- text/css(css文件)
- text/javascript(js文件)
- application/x-www-form-urlencoded(普通的表单提交)
- multipart/form-data(文件提交)
- application/json(json传输)
- application/xml(xml文件)
关于响应(Response)
视图函数的返回值会被自动转换为一个响应对象,Flask的转换逻辑如下:
-
如果返回的是一个合法的响应对象,则直接返回。
-
如果返回的是一个字符串,那么Flask会重新创建一个werkzeug.wrappers.Response对象,Response将该字符串作为主体,状态码为200,MIME类型为text/html,然后返回该Response对象。
-
如果返回的是一个元组,元祖中的数据类型是(response,status,headers)。status值会覆盖默认的200状态码,headers可以是一个列表或者字典,作为额外的消息头。
-
如果以上条件都不满足,Flask会假设返回值是一个合法的WSGIt应用程序,并通过Response.force_type(rv,request.environ)转换为一个请求对象。
以下将用例子来进行说明:
第一个例子:直接使用Response创建:
from werkzeug.wrappers import Response
@app.route('/about/')
def about():
resp = Response(response='about page',status=200,content_type='text/html;charset=utf-8')
return resp
第二个例子:可以使用make_response函数来创建Response对象,这个方法可以设置额外的数据,比如设置cookie,header信息等:
from flask import make_response
@app.route('/about/')
def about():
return make_response('about page')
第三个例子:通过返回元组的形式:
@app.errorhandler(404)
def not_found():
return 'not found',404
第四个例子:自定义响应。自定义响应必须满足三个条件:
-
必须继承自Response类。
-
实现类方法force_type(cls,rv,environ=None)。
-
必须指定app.response_class为你自定义的Response
以下将用一个例子来进行讲解,Restful API都是通过JSON的形式进行传递,如果你的后台跟前台进行交互,所有的URL都是发送JSON数据,那么此时你可以自定义一个叫做JSONResponse的类来代替Flask自带的Response类:
from flask import Flask,jsonify
from werkzeug.wrappers import Response
app = Flask(__name__)
class JSONResponse(Response):
default_mimetype = 'application/json'
@classmethod
def force_type(cls,response,environ=None):
if isinstance(response,dict):
response = jsonify(response)
return super(JSONResponse,cls).force_type(response,environ)
app.response_class = JSONResponse
@app.route('/about/')
def about():
return {"message":"about page"}
if __name__ == '__main__':
app.run(host='0.0.0.0',port=8000)
此时如果你访问/about/这个URL,那么在页面中将会显示:
{
"message": "about page"
}
注意以上例子,如果不写app.response_class = JSONResponse,将不能正确的将字典返回给客户端。因为字典不在Flask的响应类型支持范围中,那么将调用app.response_class这个属性的force_type类方法,而app.response_class的默认值为Response,因此会调用Response.force_class()这个类方法,他有一个默认的算法转换成字符串,但是这个算法不能满足我们的需求。因此,我们要设置app.response_class=JSONResponse,然后重写JSONResponse中的force_type类方法,在这个方法中将字典转换成JSON格式的字符串后再返回。