zoukankan      html  css  js  c++  java
  • flask请求钩子、HTTP响应、响应报文、重定向、手动返回错误码、修改MIME类型、jsonify()方法 --

    请求钩子:

    当我们需要对请求进行预处理和后处理时,就可以用Flask提供的回调函数(钩子),他们可用来注册在请求处理的不同阶段执行的处理函数。这些请求钩子使用装饰器实现,通过程序实例app调用,以 before_request钩子为例(请求之前),当你对一个函数附加了app.before_request装饰器后,就会将这个函数注册为before_request处理函数,每次执行请求前都会触发所有before_request函数

    Flask默认 实现的五种钩子:

     

    示例代码:

    定义了三个视图函数 A,B,C,其中C使用了after_this_request钩子,在网页中请求url 为/C,会触发C视图函数,在执行该函数的流程是先进入before_first_requst钩子,然后进入before_request钩子,之后进入C函数内部注册的after_this_request钩子函数。

    之后的after_requet和teardown_request在触发响应条件时会执行。

    @app.before_request
    def before_request():
        print "before request"

    @app.before_first_request
    def before_first_request():
        print "before first request"

    # @app.after_request
    # def after_request(response):
    #     print "after request"

    @app.after_request
    def per_request_callbacks(response):
        for func in getattr(g, 'call_after_request', ()):
            print "after request"
           
    response = func(response)
        return response

    @app.route('/A')
    def A():
        return '<p> this is function A</p>'

    @app.route('/B')
    def B():
        return '<p>this is function B</p>'

    @app.route('/C')
    def C():
        @flask.after_this_request
        def after_this_request(response):
            print "after this request"
           
    #raise ValueError
           
    return response
        return "<p>after this request</p>"



    if __name__ == '__main__':
        app.run(debug = True)

    结果:

     

    http响应

    Flask程序中,客户端的请求会触发相应的视图函数,获取返回值作为响应的主题,最后生成完整的响应,即响应报文

    响应报文:

    响应报文有协议版本、状态码、原因短语、响应首部和响应主体组成

    响应报文的首部包含一些关于响应和服务器的信息,这些内容有Flask生成,而我们在视图函数中返回的内容即为响应报文中的主体内容。浏览器接收响应后,会把返回的响应主题解析并显示 在浏览器窗口上。

    HTTP状态码用来表示请求处理的结果

     

    在Flask中生成响应

    响应在Flask中使用Response对象表示 ,响应报文中的大部分内容有服务器处理,大多数情况下,我们只负责返主题内容。

    Flask处理请求时会先判断是否可以找到与请求URL相匹配的路由,如果没有则返回404响应。如果有,则调用对应的视图函数,视图函数的返回值构成了响应报文的主题内容,正确返回时状态码默认为200,Flask会调用make_response()方法将视图函数返回值转换为响应对象。

    完整的说,视图函数可以返回最多有三个元素组成的元祖:响应主题、状态码、首部字段。其中首部字段可以为字典,或是两元素元祖组成的列表([(‘Location’,’http://localhost:5000/hi’),(‘contentType’,’…’)])

    flask请求成功时默认的响应码是200

    可以对请求指定不同的响应状态码:指定url为hi的请求的响应状态码为201

    结果:

    重定向

    重定向可以通过修改30X响应的首部Location的字段的值来进行:

    在视图函数中,return语句后面用字典的形式指定Location的值,访问的时候,会根据Location的值进行重定向

    @app.route('/redirect')
    def redirect():
        return "<p>This is a redirect pare</p>"

    @app.route('/hello1')
    def hello1():
        print "redirect..."
       
    return '', 302, {'Location': 'http://127.0.0.1:5000/redirect'}
    if __name__ == '__main__':
        app.run(debug = True)

    结果:

     

    访问hello1路径时,页面进行了重定向,重定向的位置是首部中Location字段的URL

    也可以通过Flask提供的redirect()函数来生成重定向响应

    from flask import Flask,redirect

    @app.route('/hello2')
    def hello2():
        return redirect('http://localhost:5000/redirect')

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

    修改重定向redirect函数返回码为303

    @app.route('/hello2')
    def hello2():
        return redirect('http://localhost:5000/redirect',303)

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

     

    程序内重定向到其他视图函数

    如果要在程序内重定向到其他视图函数,可以在redirect()函数中使用url_for()函数生成目标URL,可以在响应首部的Location看到重定向的目标URL,重定向到其他视图其实就是重定向到该视图对应的url

    from flask import Flask,redirect,url_for

    @app.route('/hello3')
    def hello3():
        return redirect(url_for('redirect'))#重定向到/redirect

    @app.route('/redirect')
    def redirect():
        return "<p>This is a redirect pare</p>"

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

    手动返回错误响应

    大多数情况下,Flask会自动处理常见的错误响应。HTTP错误对应的异常类在Werkzeug的werkzeug.exceptions模块中定义,抛出这些异常即可返回对应的错误响应。如果你想手动返回错误响应,可以使用Flask提供的abort()函数。

    在abort()函数中传入状态码即可返回对应的错误响应。

    需要注意,abourt()函数被调用后,abort()函数之后的代码不会被执行。

    例子为手动返回404错误:

    from flask import abort

    @app.route('/404')
    def not_found():
        abort(404)



    if __name__ == '__main__':
        app.run(debug = True)

    结果:

    响应格式:

    在http响应中,数据可以通过多种格式传输,我们会使用HTML格式,这也是Flask中的默认审核制。在特定的情况下,我们也会用其他格式。不同的响应数据格式需要设置不同的MIME类型,MIME类型在首部的Content-Type字段中定义,以默认的HTML类型为例 :

    Content-Type:  text/html; charset=utf-8

    MIME类型

    MIME类型(又称media type或content type)是一种用来表示文件类型的机制,它与文件扩展名相对应,可以让客户端区分不同的内容类型,并执行不同的操作。一般的格式为“类型名/子类型名”。其中子类型名一般为文件扩展名。比如HTML的MIME类型为”text/html”,png图片的MIME类型为”image/png”。

    如果想使用其他MIME类型,可以通过Flask提供的make_response()方法生成响应对象,传入响应的主体作为参数,然后使用响应对象的mimetype属性设置MIME类型,不需要设置字符集(charset)选项。

    修改MIME类型:

    from flask import make_response

    @app.route('/foo')
    def foo():
        response = make_response('Hello,World!')
        response.mimetype = 'text/plain'
       
    return response

    结果:

    常用的数据格式(纯文本、HTML、XML和JSON)对应的MIME类型

    纯文本的MIME类型:text/plain

    HTML的MIME类型:text/html

    XML的MIME类型:application/xml

    JSON的MIME类型:application/json

    Flask提供JSON的支持

    Flask通过引入Python标准库中的json模块为程序提供了JSON支持。你可以直接从Flask中导入json对象,然后调用dumps()方法将字典、列表或元祖序列化为JSON字符串,在使用前面介绍的方法修改MIME类型,即可返回JSON响应

    例如:

    用json模块把响应转换为JSON

    from flask import Flask,make_response,json

    @app.route('/foo1')
    def foo1():
        data={'name':'Sam Xia','gender':'male'}
        response = make_response(json.dumps(data))
        response.mimetype = 'application/json'
       
    return response

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

     

    也可以jsonnify()方法转换响应内容为JSON

    不过我们一般并不直接使用json模块的dumps()、load()等方法,因为Flask通过包装这些方法提供了更方便的jsonify()函数。用jsonify()函数,我们进需要传入数据或参数,他会把传入的参数转化成JSON字符串作为响应的主体,然后生成一个响应对象,并且设置正确的MIME类型。

    例如:

    @app.route('/foo3')
    def foo3():
        return jsonify(name = 'Sam Xia',gender = 'make')

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

    jsonify()函数可以接收多种形式的参数。可以传入如上边的关键字参数,也可以像dumps()方法一样传入字段、列表或元祖,如:


    from flask import jsonify

    @app.route('/foo3')
    def foo3():
        return jsonify({'name':'Sam Xia','gender' : 'make'})#传入字典

    if __name__ == '__main__':
        app.run(debug = True)

    结果:

    jsonify()函数可以自定义响应码

    from flask import jsonify

    @app.route('/foo3')
    def foo3():
        return jsonify(message = 'Error !'),500
    if __name__ == '__main__':
        app.run(debug = True)

    结果:

     

  • 相关阅读:
    EasyPlayer RTSP Windows播放器D3D,GDI的几种渲染方式的选择区别
    EasyPlayer RTSP Windows播放器D3D,GDI的几种渲染方式的选择区别
    libEasyPlayer RTSP windows播放器SDK API接口设计说明
    libEasyPlayer RTSP windows播放器SDK API接口设计说明
    EasyPlayer windows RTSP播放器OCX插件使用说明
    EasyPlayer windows RTSP播放器OCX插件使用说明
    EasyDSS RTMP流媒体服务器是怎样炼成的:Easy而且更加互联网!
    EasyDSS RTMP流媒体服务器是怎样炼成的:Easy而且更加互联网!
    关于EasyRTSPClient、EasyPlayer RTSP流重连问题的解释
    关于EasyRTSPClient、EasyPlayer RTSP流重连问题的解释
  • 原文地址:https://www.cnblogs.com/xiaxiaoxu/p/10389608.html
Copyright © 2011-2022 走看看