欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

django源码分析 LazySetting对象

程序员文章站 2022-03-20 20:03:12
一、django中通过LazySetting对象来获取项目的配置,LazySetting对象有什么特性?为什么使用这个对象? LazySetting顾名思义,就是延迟获取配置内容。比如,我们定义了一个对象A,并对其添加了一些属性,对A初始化时,我们将A的属性的值设置为空,当我们要访问A其中的一个属性 ......

一、django中通过lazysetting对象来获取项目的配置,lazysetting对象有什么特性?为什么使用这个对象?

lazysetting顾名思义,就是延迟获取配置内容。比如,我们定义了一个对象a,并对其添加了一些属性,对a初始化时,我们将a的属性的值设置为空,当我们要访问a其中的一个属性时,此时属性的值为空,我们才加载属性的值,并将空值设置为对应的值,返回属性值,下次获取属性值时,属性值不为空,直接返回属性值。

为什么要使用lazysetting? 

django项目在初始化的时候, 通过lazysetting,我们就可以在django获取某个配置的值之前,将配置的值先自定义为某个值,django再去获取该配置的值的时候,配置已经有了值,直接返回该配置的值。

二、django是如何实现lazysetting对象的?

1. 在说lazysetting对象之前,我们先看一下python的类属性的查找方式:

在查找一个实例化的类属性的时候

  1. 首先查找这个类的实例属性是否存在,存在直接返回
  2. 如果类的实例属性中不存在,则在类的类属性中查找,类属性中存在,则返回
  3. 如果类属性中也不存在,若定义了__getattr__方法,则根据__getattr__方法获取属性

在python中类属性和实例属性会记录在类的一个内置变量__dict__中,类属性和实例属性有各自维护的__dict__

class a:
    a = '类属性'

    def __init__(self):
        self.a = '实例属性'

print(a.__dict__)
print(a().__dict__)

输出的结果为

{'__module__': '__main__', 'a': '类属性', '__init__': <function a.__init__ at 0x04bbab28>, '__dict__': <attribute '__dict__' of 'a' objects>, '__weakref__': <attribute '__weakref__' of 'a' objects>, '__doc__': none}
{'a': '实例属性'}

 

注意的是,类属性里面不只有a属性,还有一些其他类有关的属性

我们可以看到,类属性里面a的值和实例属性里面a的值不一样。类属性和实例属性是分别维护的。

2. 下面我们写几个实例化的类查找属性的例子

第一个例子

class a:
    a = 'aa'
    b = 'bb'

    def __init__(self):
        self.a = 'aa'

obj = a()
print(obj.a)
print(obj.b)

 

输出的结果是(先在实例属性中查找,找不到,再到类属性中查找)

aa
bb

 

如果我们print(obj.c)则会报错,因为在类属性中和实例属性中都找不到c

第二个例子

class a:
    a = 'aa'
    b = 'bb'

    def __init__(self):
        self.a = 'aa'

    def __getattr__(self, item):
        return 'cc'

obj = a()
print(obj.a)
print(obj.b)
print(obj.c)

 

输出的结果和上面类似,只是print(obj.c)不报错了,因为我们定义了__getattr__方法,实例属性中和类属性中都找不到时,就会使用这个方法获取属性

aa
bb
cc

 

3. django的lazysetting类的实现

lazysetting类的实现就是通过定义__getattr__方法实现的,lazysetting类的__getattr__源码如下

def __getattr__(self, name):
    """return the value of a setting and cache it in self.__dict__."""
    if self._wrapped is empty:
        self._setup(name)
    val = getattr(self._wrapped, name)
    self.__dict__[name] = val
    return val

 

我们在lazysetting对象中查找一个属性的时候,先在实例属性(self.__dict__)中查找,没有找到话,通过__getattr__方式获取,获取到后,将属性值保存到实例属性中,这样就实现了属性在使用的时候

再获取,然后保存。我们还可以再获取属性之前,先将属性的值自定义,这样就可以不用使用__getattr__的方式来获取默认的值。