Python __call__()方法
本节再介绍 Python 类中一个非常特殊的实例方法,即 __call__()。该方法的功能类似于在类中重载 () 运算符,使得类实例对象可以像调用普通函数那样,以“对象名()”的形式使用。
举个例子:
class PLanguage: # 定义__call__方法 def __call__(self, name, add): print("调用__call__()方法", name, add) p_obj = PLanguage() p_obj("赛兔子", "https://home.cnblogs.com/u/qingchengzi")
程序执行结果:
调用__call__()方法 赛兔子 https://home.cnblogs.com/u/qingchengzi
可以看到,通过在 PLanguage 类中实现 __call__() 方法,使的 p_obj 实例对象变为了可调用对象。
Python 中,凡是可以将()直接应用到自身并执行,都称为可调用对象。可调用对象包括自定义的函数、Python 内置函数以及本节所讲的类实例对象。
对于可调用对象,实际上“名称()”可以理解为是“名称.__call__()”的简写。仍以上面程序中定义的 p_obj实例对象为例,其最后一行代码还可以改写为如下形式:
p_obj.__call__("赛兔子", "https://home.cnblogs.com/u/qingchengzi")
再举一个自定义函数的例子:
def say(): print("Python教程:https://www.cnblogs.com/qingchengzi/p/18055134") say() say.__call__()
程序执行结果:
Python教程:https://www.cnblogs.com/qingchengzi/p/18055134
Python教程:https://www.cnblogs.com/qingchengzi/p/18055134
用__call__()弥补hasattr()函数的短板
前面章节介绍了 hasattr() 函数的用法,该函数的功能是查找类的实例对象中是否包含指定名称的属性或者方法,但该函数有一个缺陷,即它无法判断该指定的名称,到底是类属性还是类方法。
要解决这个问题,我们可以借助可调用对象的概念。要知道,类实例对象包含的方法,其实也属于可调用对象,但类属性却不是。举个例子:
class PLanguage: def __init__(self): self.name = "赛兔子" self.add = "https://home.cnblogs.com/u/qingchengzi" def say(self): print("我正在学Python") p_obj = PLanguage() if hasattr(p_obj, "name"): print(hasattr(p_obj.name, "__call__")) print("**********") if hasattr(p_obj, "say"): print(hasattr(p_obj.say, "__call__"))
程序执行结果:
False
**********
True
可以看到,由于 name 是类属性,它没有以 __call__ 为名的 __call__() 方法;而 say 是类方法,它是可调用对象,因此它有 __call__() 方法。
浙公网安备 33010602011771号