python的高级特性3:神奇的__call__与返回函数

__call__是一个很神奇的特性,只要某个类型中有__call__方法,,我们可以把这个类型的对象当作函数来使用。

也许说的比较抽象,举个例子就会明白。

In [107]: f = abs

In [108]: f(-10)
Out[108]: 10

In [109]: dir(f)
Out[109]: 
['__call__',
 '__class__',
 '__delattr__',
 '__dir__',
...]

上例中的f对象指向了abs类型,由于f对象中有__call__方法,因此f(-10)实现了对abs(-10)的重载。

 

ps:由于变量/对象/实例可以指向函数,而函数能够接受变量,因此可以看出函数可以接受另一个函数作为参数,所以__call__就实现装饰器的基础。

 

扩展部分:返回函数

函数或类一般有返回值,而python中有一个神奇的特性就是返回函数

In [134]: %cpaste
Pasting code; enter '--' alone on the line to stop or use Ctrl-D.
:
:def lazy_sum(*args):
:    def sum():
:        ax = 0
:        for n in args:
:            ax = ax + n
:        return ax
:    return sum
:--

In [135]: f = lazy_sum(1,3,5,7,9)

In [136]: f
Out[136]: <function __main__.lazy_sum.<locals>.sum>

In [137]: f()
Out[137]: 25

为什么返回函数能够这么神奇,咱们一探究竟。

In [138]: dir(f)
Out[138]: 
['__annotations__',
 '__call__',
 '__class__',
 ...
 '__getattribute__',
...
 '__setattr__',
]

查看一下type,真相打败,原来是因为f里有__call__的内建方法。

posted @ 2015-07-19 21:53  李雷雷alexkn  阅读(5445)  评论(0编辑  收藏  举报