一日一技:使用装饰器实现类属性的懒加载

简介: 一日一技:使用装饰器实现类属性的懒加载

摄影:产品经理禁止钓鱼

未闻Code

博主喜欢Python和爬虫,已经出了两本书。这里是他灵感的发源地。关注这个公众号,你的生产效率在三天内就会得到提高。P.S.: 这个公众号日更。

544篇原创内容

公众号

假设我们有一个工具类MongoUtil,它的作用是封装一些数据库操作。例如:

import pymongo
class MongoUtil:
    def __init__(self):
        connect = pymongo.MongoClient()
        db = connect.tieba
        self.post = db.post
        self.user = db.user
    def write_post(self, post):
        # 处理post信息
        self.post.insert_one(post)
    def read_user_info(self):
        rows = self.user.find()
        # 读取user信息并处理
        # ...

我们发现这样写有一个问题——类在初始化的时候,就会创建数据库的链接。但我们并不是在类刚刚初始化时就读写数据库。

为了让数据库在第一次使用时再创建连接,我们就要实现懒加载机制:

import pymongo
class MongoUtil:
    def __init__(self):
        connect = pymongo.MongoClient()
        self.db = connect.tieba
        self.post = None
        self.user = None
    def write_post(self, post):
        # 处理post信息
        if not self.post:
            self.post = self.db.post
        self.post.insert_one(post)
    def read_user_info(self):
        if not self.user:
            self.user = self.db.user
        rows = self.user.find()
        # 读取user信息并处理
        # ...

这样写确实实现了懒加载,但每一个操作都需要判断当前是否联系到了对应的集合中。这样就会出现大量的重复代码。

为了解决这个问题,我们可以使用装饰器实现一个懒加载机制:

import pymongo
class lazy:
    def __init__(self, func):
        self.func = func
    def __get__(self, instance, cls):
        if instance is None:
            return self
        else:
            value = self.func(instance)
            setattr(instance, self.func.__name__, value)
            return value
class MongoUtil:
    def __init__(self):
        connect = pymongo.MongoClient()
        self.db = connect.tieba
    @lazy
    def post(self):
        return self.db.post
    @lazy
    def user(self):
        return self.db.user
    def write_post(self, post):
        # 处理post信息
        self.post.insert_one(post)
    def read_user_info(self):
        rows = self.user.find()
        # 读取user信息并处理
        # ...

我们实现了一个装饰器类lazy来装饰两个类属性postuser。当self.post第一次被调用时,它会正常连接结合,当第二次或以上访问self.post时,就会直接使用第一次返回的对象,不会再次连接MongoDB的集合。self.user同理。

我们来测试一下,如下图所示。

可以看到,第二次调用self.post时,并没有打印出第一次访问self.post,因为第二次会直接使用之前的缓存。

最后,特别说明:本文使用MongoDB举例只是为了说明基于装饰器的类属性懒加载的代码写法。而实际上,pymongo已经自动实现了懒加载机制,当我们直接db.tieba.post时,它并不会真的去连接MongoDB,只有当我们要增删改查集合里面的数据时,pymongo才会创建连接。

目录
相关文章
|
缓存 NoSQL MongoDB
一日一技:使用装饰器实现类属性的懒加载
一日一技:使用装饰器实现类属性的懒加载
77 0
|
10月前
|
JavaScript
03_装饰器
03_装饰器
79 1
|
7月前
|
测试技术 Python
装饰器
【8月更文挑战第1天】
39 2
|
JavaScript
你以为装饰器那么容易学吗?
最近在使用mobx的时候考虑到每个action都要各种try...catch,各种toast loading,觉得好烦,然后想到ES7的装饰器不就是为了少写代码的吗?于是我就立马付诸实践,开始我的装饰器的第一个应用,结果踩了好多坑,瞬间感觉说的跟做得果然不一样!
1346 0
|
JSON 数据格式
装饰器的实际应用
使用装饰器模式改造slf4j打印json格式日志
788 0
装饰器的实际应用
|
JavaScript 前端开发
【JS精粹】原型链继承和构造函数继承的 “毛病”
先从面向对象讲起,本瓜认为:面向对象编程,它的最大能力就是:复用! 咱常说,面向对象三大特点,封装、继承、多态。
|
JavaScript 前端开发 Java
JavaScript创建对象(四)——组合使用构造函数和原型模式
在JavaScript创建对象(三)——原型模式中,我们阐述了原型模式存在的两个问题:一是没办法通过构造函数初始化对象属性,二是共享引用类型的数据导致数据错乱。
1113 0
|
10月前
|
JavaScript 前端开发 编译器
TypeScript 类 第二章 【参数属性,存取器,静态属性】
TypeScript 类 第二章 【参数属性,存取器,静态属性】
105 2