51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 2663|回复: 1
打印 上一主题 下一主题

[转贴] 一日一技:在Python里面实现链式调用

[复制链接]
  • TA的每日心情
    擦汗
    1 小时前
  • 签到天数: 1047 天

    连续签到: 5 天

    [LV.10]测试总司令

    跳转到指定楼层
    1#
    发表于 2021-9-15 09:57:28 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
    我们在使用Django的models查询数据库时,可以看到有这种写法:
    1. form app.models import XXX  
    2.   query = XXX.objects.all()  
    3.   query = query.filter(name=123, age=456).filter(salary=999)  
    复制代码
     在这种写法里面,query对象有一个filter方法,这个方法的返回数据还可以继续调用filter方法,可以这样无限制地调用下去。
      这种写法是怎么实现的呢?
      如果我们直接写一个类的方法,看看能不能这样调用:

    1. class Query:  
    2.   def filter(self):  
    3.   pass  
    4.   query = Query()  
    5.   query.filter().filter()  
    复制代码

    直接对query.filter()返回的结果再调用一次filter,就会导致报错了。这是因为在没有显式写return语句的时候,方法会返回None,而None对象是没有所谓的filter方法的。
      那么什么东西有filter方法呢?显然我们的query对象有filter方法。那么如何让这个方法返回自身这个对象呢?
      这个时候,我们就要看看我们在定义类方法的时候,总会写的的第一个参数self了。几乎每个类方法里面都会有它。大家只知道在类里面调用类方法的时候可以用self.xxx(),在调用类属性的时候可以用self.yy,那么有没有思考过,这个东西如果单独使用会怎么样呢?
      实际上,self指的就是这个类实例化成一个对象以后,这个对象自身。而这个对象显然是有filter方法的。所以我们修改一下filter方法,让它返回self:
    1. class Query:  
    2.       def filter(self):  
    3.           return self  
    4.   query = Query()  
    5.   query.filter().filter()  
    复制代码

    从图中可以看出,现在已经不会报错了。那么回到最开始的问题,Django里面的链式调用传入查询参数是如何实现的呢?
      实际上这里涉及到一个惰性查询的问题。
      当我们不停调用.filter()方法的时候,Django会把这些查询条件全部缓存起来,只有当我们需要获取结果,或者查询满足条件的数据有多少条时,它才会真正地连接数据库去查询。
      所以我们这里要模拟这个环境,把查询条件缓存起来。
      那么为了获取调用方法时传入的参数名,我们就要使用**kwargs参数。这个参数可以接受所有的key=value形式的参数:
    1. class Query():  
    2.       def __init__(self):  
    3.           self.query_condition = {}  
    4.   
    5.       def filter(self, **kwargs):  
    6.           self.query_condition.update(kwargs)  
    7.           return self  
    8.   
    9.   query = Query()  
    10.   a = query.filter(name='kingname').filter(age__gt=15, address='yyyyyy').filter(salary=99999)  
    11.   print(query.query_condition)  
    复制代码
    运行效果如下图所示:

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


    本帖子中包含更多资源

    您需要 登录 才可以下载或查看,没有帐号?(注-册)加入51Testing

    x
    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-11-15 10:53 , Processed in 0.067165 second(s), 25 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表