Django-admin单例模式和懒加载

Django-admin单例模式和懒加载

单例模式

python 复制代码
class Foo:
    def __init__(self):
        self.name = "张三"
        
    def __new__(cls, *args, **kwargs):
        empty_object = super().__new__(cls)
        return empty_object

obj1 = Foo()
obj2 = Foo()

当我们实例化对象时,就会在内存开一个空间

python的执行顺序是:

  • 调用__new__方法创建一个空对象
  • 调用__init__方法向空对象中赋值name="张三"
  • 所以称__new__方法为构造方法,__init__方法为初始化方法

单例模式的目的,就是让我们在创建类对象时,都使用第一次创建的类对象,而不是像上文那样每次使用时都创建一个对象

例如:

python 复制代码
import admin	# 1

admin.site		# 2

import admin	# 3

admin.site		# 4

python的执行顺序是

  • 1加载admin.py文件

  • 2实例化一个对象

  • 3当我们再次导入admin.py文件时,python不会重新加载,步骤4也就是一开始创建的admin对象,并不会重新创建一个新的对象

如何实现一个单例模式

python 复制代码
class Foo:
    instance = None

    def __init__(self):
        self.name = "张三"

    def __new__(cls, *args, **kwargs):
        if cls.instance:
            return cls.instance
        cls.instance = empty_object = super().__new__(cls)
        return empty_object


obj1 = Foo()
obj2 = Foo()

这样如果还没有创建实例化的类,instance=None,创建并同时将创建的类赋值给instance

如果已经创建对象,即instance=对象,就直接返回这个对象

可以看出两个对象的内存地址是一样的

python实例化创建对象时,不直接创建对象,而是先创建一个代理(不确定是否会使用实例化的功能)

当我们真正调用实例化的功能的时候,再真正创建对象

懒加载

在python中,我们实例化对象后,可以拿到实例化类中的变量,也可以修改

python 复制代码
class Info:
    def __init__(self):
        self.name = "张三"
        self.age = 999

obj = Info()
print(obj.name)
obj.name = "李四"
print(obj.name)

当我们访问一个实例化类中不存在的对象时,一般情况下会报错

但是如果我们在实例化的类中定义__getattr__方法,如果访问一个实例化类中不存在的对象时,会返回这个方法的return

python 复制代码
class Info:
    def __init__(self):
        self.name = "张三"
        self.age = 999

    def __getattr__(self, item):
        return "访问了不存在的类变量"


obj = Info()
print(obj.xxxxxxxxxxx)

django-admin如何实现单例模式和懒加载

django-admin源码中实例化site对象时,并不会真正实例化创建这个对象

当它调用register方法时,并没有这个方法,就会执行它的父类LazyObject中的__getattr__方法

而这里的__getattr__方法实际上时执行new_method_proxy方法

我们需要注意,父类LazyObject中有一个类变量_wrapped默认为None

当我们执行new_method_proxy方法时,会执行_setup()方法并将它赋值给_wrapped

这样当下次实例化对象时,由于_wrapped不为空,就会直接返回,不会再次创建对象,这就实现了单例模式

self._wrapped其实就是真正的实例化对象,这就实现了懒加载

相关推荐
喵手22 分钟前
Python爬虫实战:用代码守护地球,追踪WWF濒危物种保护动态!
爬虫·python·爬虫实战·濒危物种·零基础python爬虫教学·wwf·濒危物种保护动态追踪
梦想的旅途223 分钟前
如何通过 QiWe API 实现企业微信主动发消息
开发语言·python
喵手28 分钟前
Python爬虫实战:自动化抓取 Pinterest 热门趋势与创意!
爬虫·python·爬虫实战·pinterest·零基础python爬虫教学·采集pinterest热门趋势·热门趋势预测
凌晨一点的秃头猪34 分钟前
Python文件操作
开发语言·python
小张贼嚣张43 分钟前
数据分析全流程实战:Python(Pandas/Matplotlib/Numpy)+ MySQL(附可下载数据源+多图形绘制)
python·数据分析·pandas
努力的小白o(^▽^)o1 小时前
面向课堂考勤场景的桌面端人脸识别签到系统
python·人脸识别
sa100272 小时前
淘宝商品详情 API 接口开发实战:item_detail 调用、参数与 Python 示例
linux·数据库·python
Dapenson2 小时前
腾讯小龙虾WorkBuddy技能与插件深度解析
python·ai
无心水2 小时前
【常见错误】2、Java并发编程避坑指南:从加锁失效到死锁,10个案例教你正确使用锁
java·开发语言·python
困死,根本不会3 小时前
Python 连接 iBeacon 蓝牙设备超详细学习笔记
python·蓝牙服务·ibeacon