前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一日一技:在Python里面实现链式调用

一日一技:在Python里面实现链式调用

作者头像
青南
发布2019-07-15 14:43:00
2.6K0
发布2019-07-15 14:43:00
举报
文章被收录于专栏:未闻Code未闻Code
我们在使用Django的models查询数据库时,可以看到有这种写法:
form app.models import XXXquery = XXX.objects.all()query = query.filter(name=123, age=456).filter(salary=999)

在这种写法里面,query对象有一个 filter方法,这个方法的返回数据还可以继续调用 filter方法,可以这样无限制地调用下去。

这种写法是怎么实现的呢?

如果我们直接写一个类的方法,看看能不能这样调用:

class Query:    def filter(self):        passquery = Query()query.filter().filter()

直接对 query.filter()返回的结果再调用一次 filter,就会导致报错了。这是因为在没有显式写return语句的时候,方法会返回 None,而None对象是没有所谓的 filter方法的。

那么什么东西有 filter方法呢?显然我们的query对象有 filter方法。那么如何让这个方法返回自身这个对象呢?

这个时候,我们就要看看我们在定义类方法的时候,总会写的的第一个参数 self了。几乎每个类方法里面都会有它。大家只知道在类里面调用类方法的时候可以用 self.xxx(),在调用类属性的时候可以用 self.yy,那么有没有思考过,这个东西如果单独使用会怎么样呢?

实际上, self指的就是这个类实例化成一个对象以后,这个对象自身。而这个对象显然是有 filter方法的。所以我们修改一下 filter方法,让它返回 self:

class Query:    def filter(self):        return selfquery = Query()query.filter().filter()

从图中可以看出,现在已经不会报错了。那么回到最开始的问题,Django里面的链式调用传入查询参数是如何实现的呢?

实际上这里涉及到一个 惰性查询的问题。

当我们不停调用 .filter()方法的时候,Django会把这些查询条件全部缓存起来,只有当我们需要获取结果,或者查询满足条件的数据有多少条时,它才会真正地连接数据库去查询。

所以我们这里要模拟这个环境,把查询条件缓存起来。

那么为了获取调用方法时传入的参数名,我们就要使用 **kwargs参数。这个参数可以接受所有的key=value形式的参数:

class Query():    def __init__(self):        self.query_condition = {}    def filter(self, **kwargs):        self.query_condition.update(kwargs)        return selfquery = Query()a = query.filter(name='kingname').filter(age__gt=15, address='yyyyyy').filter(salary=99999)print(query.query_condition)

运行效果如下图所示:

在真正需要输出结果的时候,再使用这些缓存的条件,去数据库中查询结果即可。

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

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

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档