zoukankan      html  css  js  c++  java
  • Flask中current_app和g对象

    Flask中current_app和g对象

    Flask中有两种上下文,请求上下文和应用上下文。

    请求上下文(request context)

    request和session都属于请求上下文对象。

    request:封装了HTTP请求的内容,针对的是http请求。举例:user = request.args.get('user'),获取的是get请求的参数。

    session:用来记录请求会话中的信息,针对的是用户信息。举例:session['name'] = user.id,可以记录用户信息。还可以通过session.get('name')获取用户信息。

    应用上下文(application context)

    current_app和g都属于应用上下文对象。

    current_app:表示当前运行程序文件的程序实例。

    g:处理请求时,用于临时存储的对象,每次请求都会重设这个变量。比如:我们可以获取一些临时请求的用户信息。

    • 当调用app = Flask(name)的时候,创建了程序应用对象app;
    • request 在每次http请求发生时,WSGI server调用Flask.call();然后在Flask内部创建的request对象;
    • app的生命周期大于request和g,一个app存活期间,可能发生多次http请求,所以就会有多个request和g。
    • 最终传入视图函数,通过return、redirect或render_template生成response对象,返回给客户端。

    区别: 请求上下文:保存了客户端和服务器交互的数据。 应用上下文:在flask程序运行过程中,保存的一些配置信息,比如程序文件名、数据库的连接、用户信息等。

    上下文对象的作用域

    在flask项目中某一个功能中会有多个视图,那么from flask import request,current_app,session,g,怎么保证某次请求的上下文不会被别的视图拿走呢?

    从pycharm中进入globals.py:

    _request_ctx_stack = LocalStack()
    _app_ctx_stack = LocalStack()
    current_app = LocalProxy(_find_app)
    request = LocalProxy(partial(_lookup_req_object, 'request'))
    session = LocalProxy(partial(_lookup_req_object, 'session'))
    g = LocalProxy(partial(_lookup_app_object, 'g'))
    

    线程有个叫做ThreadLocal的类,也就是通常实现线程隔离的类。而werkzeug自己实现了它的线程隔离类:werkzeug.local.Local。LocalStack就是用Local实现的。

    LocalStack是flask定义的线程隔离的栈存储对象,分别用来保存应用和请求上下文。

    它是线程隔离的意思就是说,对于不同的线程,它们访问这两个对象看到的结果是不一样的、完全隔离的。这是根据pid的不同实现的,类似于门牌号。

    而每个传给flask对象的请求,都是在不同的线程中处理,而且同一时刻每个线程只处理一个请求。所以对于每个请求来说,它们完全不用担心自己上下文中的数据被别的请求所修改。

    而这个LocalProxy 的作用就是可以根据线程/协程返回对应当前协程/线程的对象,也就是说

    • 线程 A 往 LocalProxy 中塞入 A
    • 线程 B 往 LocalProxy 中塞入 B

    无论在是什么地方,

    • 线程 A 永远取到得是 A,线程 B 取到得永远是 B

    此处引入:

    flask中current_app._get_current_object()与current_app有什么区别

    https://segmentfault.com/q/1010000005865632/a-1020000005865704

    查看LocalStack源码:

    def __init__(self):
            self._local = Local()
     
        def __release_local__(self):
            self._local.__release_local__()
     
        def _get__ident_func__(self):
            return self._local.__ident_func__
     
        def _set__ident_func__(self, value):
            object.__setattr__(self._local, '__ident_func__', value)
        __ident_func__ = property(_get__ident_func__, _set__ident_func__)
        del _get__ident_func__, _set__ident_func__
     
        def __call__(self):
            def _lookup():
                rv = self.top
                if rv is None:
                    raise RuntimeError('object unbound')
                return rv
            return LocalProxy(_lookup)
    

    app = Flask(__name__) 构造出一个 Flask App 时,App Context 并不会被自动推入 Stack 中。所以此时 Local Stack 的栈顶是空的,current_app 也是 unbound 状态。

    这就说明了为什么上下文需要激活状态,

    那么为什么在应用运行时不需要手动 app_context().push() 呢?

    因为 Flask App 在作为 WSGI Application 运行时,会在每个请求进入的时候将请求上下文推入。

    转载:https://blog.csdn.net/lyj20170608/article/details/79636583

  • 相关阅读:
    kali 2019 安装
    centos 7 安装
    windows 和linux系统下常用命令
    JavaScript中Object类型的定义
    《WPF编程宝典(第2版)》第3章 布局
    《C语言程序设计:现代方法(第2版)》第7章 基本类型
    《C语言程序设计:现代方法(第2版)》第6章 循环
    《C语言程序设计:现代方法(第2版)》第5章 选择语句
    《C语言程序设计:现代方法(第2版)》第4章 表达式
    《深入实践Spring Boot》第4章 提高数据库访问性能
  • 原文地址:https://www.cnblogs.com/jianjunliu/p/14322851.html
Copyright © 2011-2022 走看看