• flash框架-安装、路由、配置、请求与响应


    一、简介

    轻量级的框架,非常快速的就能把程序搭建起来

    Flask是一个基于Python开发并且依赖jinja2模板和Werkzeug WSGI服务的一个微型框架,对于Werkzeug本质是Socket服务端,其用于接收http请求并对请求进行预处理,然后触发Flask框架,开发人员基于Flask框架提供的功能对请求进行相应的处理,并返回给用户,如果要返回给用户复杂的内容时,需要借助jinja2模板来实现对模板的处理,即:将模板和数据进行渲染,将渲染后的字符串返回给用户浏览器。

    “微”(micro) 并不表示你需要把整个 Web 应用塞进单个 Python 文件(虽然确实可以 ),也不意味着 Flask 在功能上有所欠缺。微框架中的“微”意味着 Flask 旨在保持核心简单而易于扩展。Flask 不会替你做出太多决策——比如使用何种数据库。而那些 Flask 所选择的——比如使用何种模板引擎——则很容易替换。除此之外的一切都由可由你掌握。如此,Flask 可以与您珠联璧合。

    默认情况下,Flask 不包含数据库抽象层、表单验证,或是其它任何已有多种库可以胜任的功能。然而,Flask 支持用扩展来给应用添加这些功能,如同是 Flask 本身实现的一样。众多的扩展提供了数据库集成、表单验证、上传处理、各种各样的开放认证技术等功能。Flask 也许是“微小”的,但它已准备好在需求繁杂的生产环境中投入使用。

    - 和django的比较
        django:无socket,依赖第三方模块wsgi,中间件,路由系统(CBV,FBV),视图函数,ORM。cookie,session,Admin,Form,缓存,信号,序列化。。
          缺点:所有资源全部加载,造成一定资源的浪费
    
        Flask:无socket,中间件(扩展),路由系统,视图(CBV)、第三方模块(依赖jinja2),cookie,自带组件session弱爆了
          缺点:flask所有组件大部分都来自第三方,稳定性支持相对差些
    
    - tornodo 龙卷风
      异步IO非阻塞,原生websocket   ==原生支持socket,实现并发,其它两大框架都需要第三方
      里面组件太少,第三方支持的组件也太少

    二、Flask基础应用

    1、安装

    2、初始化、创建template文件夹

    from  flask import Flask
    app=Flask(__name__)
    
    @app.route('/')
    def index():
        return "ok"
    
    if __name__ == '__main__':
        app.run()

     return可返回类型:

      1、字符串

      2、html(render_template)

      3、跳转页面

      4、json数据类型

    配置

    # 配置文件,方式只能配置两种
    app.debug = True
    app.secret_key = '123'
    # 第二种,以字典的形式
    app.config['DEBUG']=True
    # 第三种,以文件的形式
    app.config.from_pyfile("settings.py")
    # 第四种以类的形式(推荐)
    app.config.from_object('settings.DevelopmentConfig')
    default_config = ImmutableDict({
        'DEBUG':                                get_debug_flag(default=False),
        'TESTING':                              False,
        'PROPAGATE_EXCEPTIONS':                 None,
        'PRESERVE_CONTEXT_ON_EXCEPTION':        None,
        'SECRET_KEY':                           None,
        'PERMANENT_SESSION_LIFETIME':           timedelta(days=31),
        'USE_X_SENDFILE':                       False,
        'LOGGER_NAME':                          None,
        'LOGGER_HANDLER_POLICY':               'always',
        'SERVER_NAME':                          None,
        'APPLICATION_ROOT':                     None,
        'SESSION_COOKIE_NAME':                  'session',
        'SESSION_COOKIE_DOMAIN':                None,
        'SESSION_COOKIE_PATH':                  None,
        'SESSION_COOKIE_HTTPONLY':              True,
        'SESSION_COOKIE_SECURE':                False,
        'SESSION_REFRESH_EACH_REQUEST':         True,
        'MAX_CONTENT_LENGTH':                   None,
        'SEND_FILE_MAX_AGE_DEFAULT':            timedelta(hours=12),
        'TRAP_BAD_REQUEST_ERRORS':              False,
        'TRAP_HTTP_EXCEPTIONS':                 False,
        'EXPLAIN_TEMPLATE_LOADING':             False,
        'PREFERRED_URL_SCHEME':                 'http',
        'JSON_AS_ASCII':                        True,
        'JSON_SORT_KEYS':                       True,
        'JSONIFY_PRETTYPRINT_REGULAR':          True,
        'JSONIFY_MIMETYPE':                     'application/json',
        'TEMPLATES_AUTO_RELOAD':                None,
    })
    Config默认配置

    三、路由配置

    1、路由

    from  flask import Flask,url_for
    app=Flask(__name__)
    app.debug=True
    
    # @app.route('/login',methods=['POST',"GET"],endpoint="sb")
    
    def login(nid):
        print(type(nid),nid)
        return "obj_login"
    
    app.add_url_rule('/login/<string:nid>',view_func=login,endpoint="sb",methods=['POST',"GET"])
    
    def index():
        real_url=url_for("sb")
        return real_url
    app.add_url_rule('/index',view_func=index,methods=['POST',"GET"])
    
    if __name__ == '__main__':
        app.run()

    #rule --->就是路由 #enpoint,取别名,如果没有就用当前函数名 #enpoint,不能重复 #methods=["POST","GET"] #view_func 就我们endpoint,指向的函数,也就请求该路由的时候,要响应函数 #app.add_url_rule(rule,‘取别名,反向解析’,login) #url_for 反向解析

    strict_slashes=True,  严格模式
    redirect_to ="/login"  重定向

    2、请求方式

    请求方式一

    class IndexView(views.View):
        methods=["POST","GET"]
        # decorators=[,,,,]
        def dispatch_request(self):
            print('Index')
            return 'Index!'

    请求方式二

    class IndexView(views.MethodView):
        def get(self):
            print('Index')
            return 'Index!,get'
        def post(self):
            return 'Index!,post'

    注意:

    #IndexView.as_view(name='index')都views.View里面的as_view的view
    #为什么要as_view(name='index'),
    #如果不指定,就都是view
    #所以必须指定
    例:
    app.add_url_rule('/index', view_func= IndexView.as_view(name='index'))
    rom  flask import Flask,url_for,views
    app=Flask(__name__)
    app.debug=True
    # class IndexView(views.View):
    #     methods=["POST","GET"]
    #     # decorators=[,,,,]
    #     def dispatch_request(self):
    #         print('Index')
    #         return 'Index!'
    
    class IndexView(views.MethodView):
        # methods=["GET"]
        # decorators=[,,,,]
        def get(self):
            print('Index')
            return 'Index!,get'
        def post(self):
            return 'Index!,post'
    
    #IndexView.as_view(name='index')都views.View里面的as_view的view
    #为什么要as_view(name='index'),
    #如果不指定,就都是view
    #所以必须指定
    
    app.add_url_rule('/index', view_func= IndexView.as_view(name='index'))
    完整版

     3、自定义路由匹配——正则

    from flask import Flask, views, url_for
    from werkzeug.routing import BaseConverter
    
    app = Flask(import_name=__name__)
    
    class RegexConverter(BaseConverter):
        """
        自定义URL匹配正则表达式
        """
        def __init__(self, map, regex):
            super(RegexConverter, self).__init__(map)
            self.regex = regex
    
        def to_python(self, value):
            """
            路由匹配时,匹配成功后传递给视图函数中参数的值
            """
            #value就正则匹配出来的结果
            print('value',value,type(value))
            return "asdasdasd"
    
        def to_url(self, value):
            """
            使用url_for反向生成URL时,传递的参数经过该方法处理,返回的值用于生成URL中的参数
            """
            val = super(RegexConverter, self).to_url(value)
            print(val)
            return val
    
    
    app.url_map.converters['regex1'] = RegexConverter
    @app.route('/index/<regex1("d+"):nid>',endpoint="sb")
    def index(nid):
        print("nid",nid,type(nid))
        print(url_for('sb', nid='888'))
        # /index/666
        return 'Index'
    
    if __name__ == '__main__':
        app.run()
    View Code

      1)导入from werkzeug.routing import BaseConverter

      2)写一个类RegexConverter,继承BaseConverter,实现__init__方法

      3)定义app.url_map.converters['regexl'] = RegexConverter

      4)在路由里面@app.route('/index/<regex("d+"):nid>'),regex正则匹配结果,返回to_python

       5)当我们做反向解析的解析的时候,我们的参数,会传递给to_url,return的结果才是我们拼接到我们路由上。

    四、模板

    通过定义函数func1,返回Markup(),是前端页面标签直接渲染(类比django的safe方法)

    相较django,flask可传多个参数。

    run.py

    #略开头
    def func1(st,st1):
        return Markup(f"<h1>jsaon-gdx{st}{st1}</h1>")
    
    def list():
        return render_template('list.html',html="<h1>jsaon-gdx</h1>",html1=func1)
    # 略尾部

    list.html

    {{html|safe}}
    {{html1("-DSB","-SB")}}  # 可传参拼接

    五、请求与响应

    1、

    2、session

    from flask import Flask,session
    
    app = Flask(__name__)
    app.secret_key="askjdaksd"
    app.config['SESSION_COOKIE_NAME']="dsb"
    # app.session_interface
    '''
    app.session_interface这里面看
    存session,
    1 调用save_session,将我们的session加密的val,读取配置文件['SESSION_COOKIE_NAME']得到key
    2 将1种的key,val存储到cookies
    
    取session
    1 获取request里面的cookies,获取里面key,这个key就是['SESSION_COOKIE_NAME'],值就是加密的值
    2 对该值进行解密
    
    '''
    
    @app.route("/")
    def index():
        session['jason']="gdx"
        return "ok"
    
    @app.route("/index1")
    def index1():
        print(session['jason'])
        return "ok1"
    
    if __name__ == '__main__':
        app.run()

    3、闪存(a页面产生信息,其他页面可以获取到,但只能获取一次

      #1 如果要用flash就必须设置app.secret_key = 'asdfasdf'

      #2 只能取一次,在取就没有了

      #3 可以通过  flash('普通信息',category="info"),对信息做分类

      #4 get_flashed_messages(with_categories=True,category_filter=("error",)),with_categories默认False,设为True携带信息名

      我们设置闪现,category_filter=("error",)进行分类信息的过滤

    from flask import Flask,flash,get_flashed_messages,request
    
    app = Flask(__name__)
    app.secret_key = 'asdfasdf'
    
    @app.route('/index1')
    def index():
        #(category="message", message))
        flash('超时错误',category="error")
        flash('普通信息',category="info")
        return "ssdsdsdfsd"
        # return redirect('/error')
    
    @app.route('/error1')
    def error1():
        return "ok"
    
    @app.route('/error')
    def error():
        data = get_flashed_messages(with_categories=True,category_filter=("error","info"))
        data1 = get_flashed_messages( category_filter=("error", "info"))
        print("data1",data1) # data1 [('error', '超时错误'), ('info', '普通信息')]
        print("data",data) # data ['超时错误', '普通信息']
        return "错误信息"
    
    if __name__ == '__main__':
        app.run()

    4、请求拓展

      1)before_request、after_request

    Flask请求扩展也可以写多个
    但是要注意,多个请求扩展的执行顺序是有差别的:
    对于before_request,是按照写的代码的顺序从上到下的顺序正序执行的
    对于after_request, 是按照写的代码的顺序从下到上的顺序倒序执行的
    
    如果before_request return了(即程序被拦截了),其他before_request就不执行了,但是所有的after_request都会继续执行

      2)定制错误信息 errorhandler(404)

      3)模板中定制方法(定制模板方法): template_global()  和  template_filter()

    5、中间件

    from flask import Flask
    app = Flask(__name__)
    
    @app.route("/login", methods=['GET', 'POST'])
    def index():
        pass
    
    class Md(object):
        def __init__(self, old_wsgi_app):
            self.old_wsgi_app = old_wsgi_app
    
        def __call__(self, environ, start_response):
            print("开始之前")
            ret = self.old_wsgi_app(environ, start_response)
            print("结束之后")
            return ret
    
    if __name__ =="__main__":
        app.wsgi_app = Md(app.wsgi_app) # 相当于把wsgi_app给更新了
        app.run()
     
    @app.route("/index",strict_slashes=True,redirect_to ="/login")
  • 相关阅读:
    次小生成树
    [bzoj5329] P4606 [SDOI2018]战略游戏
    CF487E Tourists
    P3225 [HNOI2012]矿场搭建
    CF #636 (Div. 3) 对应题号CF1343
    P3469 [POI2008]BLO-Blockade
    大假期集训模拟赛12
    大假期集训模拟赛11
    大假期集训模拟赛10
    小奇画画——BFS
  • 原文地址:https://www.cnblogs.com/xiaowangba9494/p/11839154.html
Copyright © 2020-2023  润新知