回調函數:把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用來調用其所指向的函數時,我們就說這是回調函數。回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用於對該事件或條件進行響應。
通俗理解就是:把一個函數作為參數傳給另一個函數,第一個函數稱為回調函數。這個被傳入的參數其實是函數指針,即指向一個函數的指針(地址)。
軟件模塊之間總是存在着一定的接口,從調用方式上,可以把他們分為三類: 同步調用、回調和異步調用。
- 同步調用是一種阻塞式調用,調用方要等待對方執行完畢才返回,它是一種單向調用;
- 回調是一種雙向調用模式,也就是說,被調用方在接口被調用時也會調用對方的接口;
- 異步調用是一種類似消息或事件的機制,不過它的調用方向剛好相反,接口的服務在收到某種訊息或發生某種事件時,會主動通知客戶方(即調用客戶方的接口.
回調和異步調用的關系非常緊密,通常我們使用回調來實現異步消息的注冊,通過異步調用來實現消息的通知。同步調用是三者當中最簡單的,而回調又常常是異步調用的基礎,因此,下面我們着重討論回調機制在不同軟件架構中的實現
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_)