Python 代码阅读(第 43 篇):构造组合函数
Python 代码阅读合集介绍:为什么不推荐Python初学者直接看项目源码
本篇阅读的代码实现了构造将输入函数依次调用的组合函数。
本篇阅读的代码片段来自于30-seconds-of-python。
compose
compose
函数接收多个函数,返回将这些函数从右至左依次调用的组合函数。接收的函数中,只有最右边的函数(组合函数中第一个调用的函数)可以接收多于一个参数,其他函数只能接收一个参数。
函数使用reduce
将一个lambda
表达式依次作用于接收的函数列表fns
。reduce
中接收两个参数的lambda
表达式返回一个新的匿名函数(lambda
表达式)。通过reduce
函数的迭代,将会把所有的fns
中的函数依次调用组合成一个函数。假设fns
为(a,b,c,d,e)
,那么reduce
函数返回的合成函数为a(b(c(d(e(*arg)))))
下面我们深入compose
函数的实现
函数compose(*fns)
接收任意数量的参数,所有这些参数会被包含在一个名为fns
的元组里。
functools.reduce(function, iterable[, initializer])
将两个参数的 function 从左至右积累地应用到iterable
的条目,以便将该可迭代对象缩减为单一的值。 例如,reduce(lambda x, y: x+y, [1, 2, 3, 4, 5])
是计算((((1+2)+3)+4)+5)
的值。 左边的参数x
是积累值而右边的参数y
则是来自iterable
的更新值。
compose
函数中reduce
的匿名函数lambda f, g: lambda *args: f(g(*args))
可以写成如下形式:
假设fns
为(a,b,c,d,e)
,reduce
函数累积的调用function1
构造函数function2
。a,b
首先作为function1
的入参,构造出运算内容是a(b(*arg))
的函数function2
。本次迭代构造出的function2
和c
作为入参调用function1
函数进行第二次迭代,function1(function2,c)
将构造运算内容是function2(c(*arg))
的函数,解开上一个迭代的function2
,新函数的运算内容是a(b(c(*arg)))
。如此依次迭代,最后获得的函数的实际运算内容是a(b(c(d(e(*arg)))))
。
compose_right
和上文compose
函数一样的原理,只需要将reduce
函数中的调用顺序交换,就可以构造一个反方向,从左至右依次调用的组合函数。
版权声明: 本文为 InfoQ 作者【Felix】的原创文章。
原文链接:【http://xie.infoq.cn/article/057e442c576fd0e086687cb30】。
本文遵守【CC-BY 4.0】协议,转载请保留原文出处及本版权声明。
评论