用Python实现链式调用

嗨喽,大家好呀~这里是爱看美女的茜茜呐

我们在使用Django的models查询数据库时,可以看到有这种写法:

dart 复制代码
form app.models import XXX
query = XXX.objects.all()
query = query.filter(name=123, age=456).filter(salary=999)

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


👇 👇 👇 更多精彩机密、教程,尽在下方,赶紧点击了解吧~

python源码、视频教程、插件安装教程、资料我都准备好了,直接在文末名片自取就可


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

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

dart 复制代码
class Query:
    def filter(self):
        pass

query = Query()
query.filter().filter()

直接对query.filter()返回的结果再调用一次filter,就会导致报错了。

这是因为在没有显式写return语句的时候,方法会返回None,而None对象是没有所谓的filter方法的。

那么什么东西有filter方法呢?

显然我们的query对象有filter方法。

那么如何让这个方法返回自身这个对象呢?

这个时候,我们就要看看我们在定义类方法的时候,总会写的的第一个参数self了。

几乎每个类方法里面都会有它。

大家只知道在类里面调用类方法的时候可以用self.xxx(),在调用类属性的时候可以用self.yy,

那么有没有思考过,这个东西如果单独使用会怎么样呢?

实际上,self指的就是这个类实例化成一个对象以后,这个对象自身。

而这个对象显然是有filter方法的。

所以我们修改一下filter方法,让它返回self:

dart 复制代码
'''
python资料获取看这里噢!! 小编 V:Pytho8987(记得好友验证备注:6 否则可能不通过)
即可获取:文章源码/教程/资料/解答等福利,还有不错的视频学习教程和PDF电子书!
'''
class Query:
    def filter(self):
        return self

query = Query()
query.filter().filter()

从图中可以看出,现在已经不会报错了。

那么回到最开始的问题,Django里面的链式调用传入查询参数是如何实现的呢?

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

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

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

那么为了获取调用方法时传入的参数名,我们就要使用**kwargs参数。

这个参数可以接受所有的key=value形式的参数:

dart 复制代码
'''
python资料获取看这里噢!! 小编 V:Pytho8987(记得好友验证备注:6 否则可能不通过)
即可获取:文章源码/教程/资料/解答等福利,还有不错的视频学习教程和PDF电子书!
'''
class Query():
    def __init__(self):
        self.query_condition = {}

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

运行效果如下图所示:

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

尾语

感谢你观看我的文章呐~本次航班到这里就结束啦 🛬

希望本篇文章有对你带来帮助 🎉,有学习到一点知识~

躲起来的星星🍥也在努力发光,你也要努力加油(让我们一起努力叭)。

最后,宣传一下呀~👇👇👇更多源码、资料、素材、解答、交流皆点击下方名片获取呀👇👇

相关推荐
小王的饲养员几秒前
Apipost 与 Apifox 数据库功能对比:深入解析与应用场景分析
大数据·数据库·人工智能·后端·postman
nextera-void11 分钟前
深入浅出 Golang:一次精神之旅
开发语言·golang·go
FE杂志社16 分钟前
全栈开发 → FastAPI碎碎念
后端·python·fastapi
Rebecca.Yan21 分钟前
Maven下载与配置&&对Java项目的理解
java·开发语言·maven
UestcXiye22 分钟前
Rust Web 全栈开发(六):在 Web 项目中使用 MySQL 数据库
数据库·rust
写不出来就跑路42 分钟前
openGauss数据库管理实战指南——基本常用操作总结
数据库·sql·gaussdb
wu27901 小时前
MYSQL笔记2
数据库·笔记·mysql
专注VB编程开发20年1 小时前
C# VB.NET多进程-管道通信,命名管道(Named Pipes)
开发语言·c#·.net
小关会打代码1 小时前
每天学习一个Python库之正则表达式库(re)
python·学习·re
Pocker_Spades_A1 小时前
飞算JavaAI:开启 Java 开发 “人机协作” 新纪元
java·开发语言·飞算javaai