参考Python回调函数用法实例
参考赶紧来学Python回调函数
1 什么是回调函数?
1.1 回调函数
回调函数就是一个被作为参数传递的函数。
在C语言中,回调函数只能使用函数指针实现。
在C++、Python、ECMAScript等更现代的编程语言中还可以使用仿函数或匿名函数。
回调函数的使用可以大大提升编程的效率,这使得它在现代编程中被非常多地使用。同时,有一些需求必须要使用回调函数来实现。
1.2 回调函数的来源
编程分为两类:系统编程(system programming)和应用编程(application programming)。
所谓系统编程,简单来说,就是编写库。而应用编程就是利用写好的各种库来编写具某种功用的程序,也就是应用。系统程序员会给自己写的库留下一些接口,即API(application programming interface,应用编程接口),以供应用程序员使用。所以在抽象层的图示里,库位于应用的底下。
当程序跑起来时,一般情况下,应用程序(application program)会时常通过API调用库里所预先备好的函数。
但是有些库函数(library function)却要求应用先传给它一个函数,好在合适的时候调用,以完成目标任务。这个被传入的、后又被调用的函数就称为回调函数(callback function)。
打个比方,有一家旅馆提供叫醒服务,但是要求旅客自己决定叫醒的方法。可以是打客房电话,也可以是派服务员去敲门,睡得死怕耽误事的,还可以要求往自己头上浇盆水。
(1)"叫醒"这个行为是旅馆提供的,相当于库函数。
(2)叫醒的方式是由旅客决定并告诉旅馆的,也就是回调函数。
(3)旅客告诉旅馆怎么叫醒自己的动作,也就是把回调函数传入库函数的动作,称为登记回调函数(to register a callback function)。
(4)回调函数通常和应用处于同一抽象层(因为传入什么样的回调函数是在应用级别决定的)。而回调就成了一个高层调用底层,底层再回过头来调用高层的过程。
回调利于模块解耦。
2 回调示例
回调机制提供了非常大的灵活性。从现在开始将图中的库函数改称为中间函数,这是因为回调并不仅仅用在应用和库之间。任何时候,只要想获得类似于上面情况的灵活性,都可以利用回调。
这种灵活性是怎么实现的呢?
乍看起来,回调似乎只是函数间的调用,但仔细一琢磨,可以发现两者之间的一个关键的不同:在回调中,利用某种方式,把回调函数像参数一样传入中间函数。可以这么理解,在传入一个回调函数之前,中间函数是不完整的。换句话说,程序可以在运行时,通过登记不同的回调函数,来决定、改变中间函数的行为。
2.1 示例一
odd奇数
even偶数
2.1.1 even.py
# 回调函数1
# 生成一个2k形式的偶数
def double(x):
return x * 2
# 回调函数2
# 生成一个4k形式的偶数
def quadruple(x):
return x * 4
2.1.2 callback_demo.py
from even import *
# 中间函数
# 接受一个生成偶数的函数作为参数,返回一个奇数
def getOddNumber(k, getEvenNumber):
return 1 + getEvenNumber(k)
if __name__ == "__main__":
k = 1
# 当需要生成一个2k+1形式的奇数时
print(getOddNumber(k, double))
# 当需要一个4k+1形式的奇数时
print(getOddNumber(k, quadruple))
# 当需要一个8k+1形式的奇数时
print(getOddNumber(k, lambda x: x * 8))
上面的代码里,给getOddNumber传入不同的回调函数,它的表现也不同,这就是回调机制的优势所在。值得一提的是,上面的第三个回调函数是一个匿名函数。
2.2 示例二
在Python中,回调函数是一种非常重要的编程概念,它允许我们将一个函数作为参数传递给另一个函数,并在需要时由另一个函数调用。回调函数的使用可以使代码更加灵活和可重用,尤其在异步编程、事件驱动编程中非常常用。
2.2.1 普通函数调用
def fn(a, b):
print(a + b)
fn(12, 34) # 46
# 若把函数名赋给了一个变量,那该变量也具备了函数的功能
demo = fn
demo(12, 34) # 46
print(type(fn)) # <class 'function'>
print(type(demo)) # <class 'function'>
2.2.2 回调函数
把一个函数(func)作为一个参数传递到另外一个函数(common)中去,那么函数func就是回调函数.
# 加法
def add(a, b):
print(a + b)
# 减法
def jian(a, b):
print(a - b)
# 乘法
def cheng(a, b):
print(a * b)
# 除法
def chu(a, b):
print(a / b)
# 封装一个函数:该函数可以实现加减乘除4中运算
def common(a, b, func):
func(a, b)
common(12, 4, add) # 16 加法,回调add函数
common(12, 4, jian) # 8 减法,回调jian函数
common(12, 4, cheng) # 48 乘法,回调cheng函数
common(12, 4, chu) # 3.0 除法,回调chu函数
2.3 中间函数的调用者
通过上面的论述可知,中间函数和回调函数是回调的两个必要部分,不过人们往往忽略了回调里的第三位主角,就是中间函数的调用者。绝大多数情况下,这个调用者可以和程序的主函数等同起来。
之所以特意强调这个第三方,是因为我在网上读相关文章时得到一种印象,很多人把它简单地理解为两个个体之间的来回调用。实际上,回调并不是"你我"两方的互动,而是ABC的三方联动。
另外,回调实际上有两种:阻塞式回调和延迟式回调。
两者的区别在于:
(1)阻塞式回调里,回调函数的调用一定发生在主函数返回之前。
(2)延迟式回调里,回调函数的调用有可能是在主函数返回之后。
另外还请注意,本文中所举的示例均为阻塞式回调。
延迟式回调通常牵扯到多线程。