python __getattr__ 与 __call__

访问类内不存在的属性就会触发__getattr__方法(前提你实现该方法)

类自己也可以作为一个方法调用,通过实现__call__

如:

class A():
    def __init__(self):
        pass
    def __call__(self):
        print('A!')

>>> A()()
A!

__str__ 是作为输出字符串时的触发方法

__repr__为调试时的输出,一般这样写:

def __str__(self):
        return self.__path
    __repr__ = __str__

看一个例子。

>>> class Chain(object):
    def __init__(self, path = ''):
        self.__path = path
        return
    def __call__(self, name):
        return Chain('%s/%s' % (self.__path, name))
    def __str__(self):
        return self.__path
    __repr__ = __str__
    def __getattr__(self, path):
        return Chain('%s/%s' % (self.__path, path))


>>> Chain().users('michael').repos
/users/michael/repos
>>> Chain().users('michael').users.repos
/users/michael/users/repos

第二个输入

这里Chain().users('michael')部分,首先代码没有定义users()方法(如果有定义方法,它的优先级高与__getattr__),所以这么把它看做未定义属性。于是触发__getattr__。__getattr__(self,path)返回Chain('/users'),相当于变成了Chain('/users')('michael').users.repos.Chain('/users')('michael')触发__call__,得到Chain('/users/michael/')。users.repos。再后面同理。最后得到Chain('/users/michael/users/repos')。通过__repr__输出。

如果是print(Chain('/users/michael/users/repos') )  则通过__str__输出。

主要Chain().users('michael')有歧义,可以把users('michael')看成方法调用(如果有定义,则优先考虑),如果没有定义也可以看成Chain().users为属性,触发__getattr____getattr__方法再返回类实例Chain().users,再触发__call__。即(Chain().users)('michael')。

参考资料:

http://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014319098638265527beb24f7840aa97de564ccc7f20f6000#0

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中,__delattr__是一个魔法方法,用于在清除实例属性时被捕获执行。通过重写__delattr__方法,我们可以在删除属性时添加自定义的逻辑。该方法的语法很简单,其中self是所有实例方法都必须有的第一个参数,属性是要删除的属性的名称。在重写该方法时,如果要实现真正的属性删除,必须调用父类的__delattr__方法,否则相关属性不会被清除。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Python __setattr__、 __getattr__、 __delattr__、__call__用法示例](https://download.csdn.net/download/weixin_38649657/12878817)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [第8.32节 Python中重写__delattr__方法捕获属性删除](https://blog.csdn.net/LaoYuanPython/article/details/96881603)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [第8.31节 Python中使用__delattr__清除属性数据](https://blog.csdn.net/LaoYuanPython/article/details/96729822)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值