目录:
- 每篇前言:
- g到底是什么?生命周期
- 在请求周期内保持数据
- 需要注意的是:
- 拓展—面向对象的私有字段
- 深入讲解一下那句:
每篇前言:
🏆🏆作者介绍:【孤寒者】—CSDN全栈领域优质创作者、HDZ核心组成员、华为云享专家Python全栈领域博主、CSDN原力计划作者
- 🔥🔥本文已收录于Flask框架从入门到实战专栏:《Flask框架从入门到实战》
- 🔥🔥热门专栏推荐:《Python全栈系列教程》、《爬虫从入门到精通系列教程》、《爬虫进阶+实战系列教程》、《Scrapy框架从入门到实战》、《Flask框架从入门到实战》、《Django框架从入门到实战》、《Tornado框架从入门到实战》、《前端系列教程》。
- 📝📝本专栏面向广大程序猿,为的是大家都做到Python全栈技术从入门到精通,穿插有很多实战优化点。
- 🎉🎉订阅专栏后可私聊进一千多人Python全栈交流群(手把手教学,问题解答); 进群可领取Python全栈教程视频 + 多得数不过来的计算机书籍:基础、Web、爬虫、数据分析、可视化、机器学习、深度学习、人工智能、算法、面试题等。
- 🚀🚀加入我一起学习进步,一个人可以走的很快,一群人才能走的更远!
g到底是什么?生命周期
在 Flask 中,g
是一个特殊的对象,它提供了一个用于存储和共享数据的全局命名空间。Flask 使用上下文全局变量来使某些对象在一个请求周期内全局可访问,而不会与其他请求的上下文冲突。g
是其中之一,专门用来存储和共享请求周期内的数据。
g
对象在每次请求开始时被重置(所以它是不能当session对象来用的!因为它不能跨请求保持状态!!!),在不同请求间是独立的,这意味着我们可以在处理请求时将数据附加到 g
,而不用担心来自其他请求的冲突或数据遗留。
在请求周期内保持数据
g
对象非常适合用来缓存数据,比如数据库连接,这样就不必在每次请求时都重新建立连接。
如下在请求开始时连接数据库,并在请求结束时关闭连接:
from flask import g, current_app
@app.before_request
def before_request():
# 假设有一个函数 get_db 来获取数据库连接
g.db = get_db()
@app.teardown_request
def teardown_request(exception=None):
# 如果数据库连接已经存在,则关闭它
db = getattr(g, 'db', None)
if db is not None:
db.close()
在上面的例子中,使用 before_request
钩子来在处理每个请求之前设置数据库连接,并将其存储在 g
对象中。然后,在请求处理完成后,使用 teardown_request
钩子来清理数据库连接。
需要注意的是:
g
对象是线程隔离的,每个请求在处理时会在其自己的线程中运行,所以不同请求中的g
对象不会相互干扰。g
对象只在请求生命周期内有效。一旦请求结束,存储在g
对象中的数据就不再可用。- 不应该使用
g
对象来存储大量数据或者对于多个请求来说应该保持不变的数据。如果需要这种类型的存储,考虑使用应用上下文、数据库或缓存系统。
拓展—面向对象的私有字段
源码LocalProxy进去:
前面我讲过上图红框那一句等价于:
self.__local = 函数
现在来细细讲一下为啥?
首先,类里面以双下划线开头的为私有属性,在类的外部访问不了会报错:
但其实Python是支持我们在外部调用的:
通过【下划线加类名加双下划线加私有属性】就可以了!
深入讲解一下那句:
在 Python 中,object.__setattr__(self, name, value)
是一种特殊的方法,用于设置属性的值。默认情况下,我们一般使用 self.name = value
的语法来设置属性,但如果需要在定义了自己的 __setattr__
方法的类中设置属性,而不触发那个类的 __setattr__
方法,就可以使用 object.__setattr__()
这种直接调用基类 object
来实现。
所以,object.__setattr__(self, "_LocalProxy__local", local)
这行代码做的就是设置 LocalProxy
实例的 __local
属性,而不会触发 LocalProxy
类已经定义的自己的 __setattr__
。
这段代码使用 Python 的 名称改写(name mangling) 来确保 __local
属性是私有的。在 Python 中,以双下划线开头的属性名称会被解释器改写,以包含类名来避免子类中的名称冲突。例如,_LocalProxy__local
实际上是 LocalProxy
类的一个私有属性。这种改写意味着它不能被类的外部直接访问,只能在类的内部访问【但是其实可以访问,但是没人会这么做,也不会有人让你这么做!】。
这样,任何对 LocalProxy
实例的 __local
属性的引用或者赋值都会通过 LocalProxy
类定义的方法进行,而不是直接操作属性,这对于保持封装和管理属性访问是很有帮助的。然而,在初始化的时候,类需要能够设置这个值,而不是通过可能已经定义的 __setattr__
方法,这就是为什么这里会直接调用 object
的 __setattr__
方法。