前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一日一技:使用装饰器实现类属性的懒加载

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

作者头像
青南
发布2019-05-27 16:57:28
6010
发布2019-05-27 16:57:28
举报
文章被收录于专栏:未闻Code未闻Code

假设我们有一个工具类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已经自动实现了懒加载机制,当我们直接 connect.tieba.post时,它并不会真的去连接MongoDB,只有当我们要增删改查集合里面的数据时,pymongo才会创建连接。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-05-24,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 未闻Code 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
云数据库 MongoDB
腾讯云数据库 MongoDB(TencentDB for MongoDB)是腾讯云基于全球广受欢迎的 MongoDB 打造的高性能 NoSQL 数据库,100%完全兼容 MongoDB 协议,支持跨文档事务,提供稳定丰富的监控管理,弹性可扩展、自动容灾,适用于文档型数据库场景,您无需自建灾备体系及控制管理系统。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档