回调函数:把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
通俗理解就是:把一个函数作为参数传给另一个函数,第一个函数称为回调函数。这个被传入的参数其实是函数指针,即指向一个函数的指针(地址)。
软件模块之间总是存在着一定的接口,从调用方式上,可以把他们分为三类: 同步调用、回调和异步调用。
- 同步调用是一种阻塞式调用,调用方要等待对方执行完毕才返回,它是一种单向调用;
- 回调是一种双向调用模式,也就是说,被调用方在接口被调用时也会调用对方的接口;
- 异步调用是一种类似消息或事件的机制,不过它的调用方向刚好相反,接口的服务在收到某种讯息或发生某种事件时,会主动通知客户方(即调用客户方的接口.
回调和异步调用的关系非常紧密,通常我们使用回调来实现异步消息的注册,通过异步调用来实现消息的通知。同步调用是三者当中最简单的,而回调又常常是异步调用的基础,因此,下面我们着重讨论回调机制在不同软件架构中的实现
import random as rd
# -----------被调用方----------------------------
def newRN(fn): # 生成10个[0,1)之间小数
ns = []
for i in range(10):
n = round(rd.random(), 2)
ns.append(n)
# 不用直接 return, 因为调用方 通知不接返回结果
# 改成回调函数方式
fn(ns) # 调用是调用方函数,这一操作称之为回调
# ----------------调用方------------------------
# 定义回调函数
def abc(*args):
# 进入到本函数内,意味着被调用方函数已执行完
print('生成数据成功')
print(args)
newRN(abc)
案例二
import random as rd
import time
def test(fn, n):
"""
测试生成 n 个随机数需要的时间(秒)
:param fn: 回调函数
:param n: 生成的数量
:return:
"""
start_time = time.time() # 开始时间
for i in range(n):
rd.random()
time.sleep(0.1) # 让CPU休眠0.1秒
end_time = time.time() # 循环结束的时间
seconds = round(end_time - start_time) # 执行的时间
fn(seconds,n) # 回传用时(秒),n 生成的数量
def test_callback(tm_s,n):
print('生成 {0}个数,用时 {1} 秒'.format(n, tm_s))
test(test_callback, 20)
案例三
def findNumbers(path, num, fn):
l = []
rowNo = 1
with open(path, 'r+') as f:
for line in f.readlines():
index = line.find(str(num))
if index != -1:
l.append((rowNo, index + 1))
rowNo += 1
fn(l)
def callback_(sl):
print(sl)
findNumbers('222.txt', 5, callback_)