徹底弄懂Python中的回調(diào)函數(shù)(callback)
摘要
主要是介紹python 的回調(diào)函數(shù)callback。
什么是回調(diào)函數(shù)
當(dāng)程序運(yùn)行是,一般情況下,應(yīng)用程序會(huì)時(shí)常通過(guò)API調(diào)用庫(kù)里所預(yù)先備好的函數(shù)。但是有些庫(kù)函數(shù)卻要求應(yīng)用先傳給它一個(gè)函數(shù),好在合適的時(shí)候調(diào)用,以完成目標(biāo)任務(wù)。這個(gè)被傳入的、后又被調(diào)用的函數(shù)就稱為回調(diào)函數(shù)(callback function)。
例如:
有一家旅館提供叫醒服務(wù),但是要求旅客自己決定叫醒的方法??梢允谴蚩头侩娫?,也可以是派服務(wù)員去敲門(mén),睡得死怕耽誤事的,還可以要求往自己頭上澆盆水。這里,“叫醒”這個(gè)行為是旅館提供的,相當(dāng)于庫(kù)函數(shù),但是叫醒的方式是由旅客決定并告訴旅館的,也就是回調(diào)函數(shù)。而旅客告訴旅館怎么叫醒自己的動(dòng)作,也就是把回調(diào)函數(shù)傳入庫(kù)函數(shù)的動(dòng)作,稱為登記回調(diào)函數(shù)(to register a callback function)
可以看到,回調(diào)函數(shù)通常和應(yīng)用處于同一抽象層(因?yàn)閭魅胧裁礃拥幕卣{(diào)函數(shù)是在應(yīng)用級(jí)別決定的)。而回調(diào)就成了一個(gè)高層調(diào)用底層,底層再回過(guò)頭來(lái)調(diào)用高層的過(guò)程。(我認(rèn)為)這應(yīng)該是回調(diào)最早的應(yīng)用之處,也是其得名如此的原因。

回調(diào)機(jī)制的優(yōu)勢(shì)
回調(diào)機(jī)制提供了非常大的靈活性。我們把圖中的庫(kù)函數(shù)改稱為中間函數(shù)了,這是因?yàn)榛卣{(diào)并不僅僅用在應(yīng)用和庫(kù)之間。任何時(shí)候,只要想獲得類似于上面情況的靈活性,都可以利用回調(diào)。有的同學(xué)可定想回調(diào)似乎只是函數(shù)間的調(diào)用,可以發(fā)現(xiàn)兩者之間的一個(gè)關(guān)鍵的不同:
在回調(diào)中,我們利用某種方式,把回調(diào)函數(shù)像參數(shù)一樣傳入中間函數(shù)??梢赃@么理解,在傳入一個(gè)回調(diào)函數(shù)之前,中間函數(shù)是不完整的。換句話說(shuō),程序可以在運(yùn)行時(shí),通過(guò)登記不同的回調(diào)函數(shù),來(lái)決定、改變中間函數(shù)的行為。這就比簡(jiǎn)單的函數(shù)調(diào)用要靈活太多了。
#回調(diào)函數(shù)1
#生成一個(gè)2k形式的偶數(shù)
def double(x):
return x * 2
#回調(diào)函數(shù)2
#生成一個(gè)4k形式的偶數(shù)
def quadruple(x):
return x * 4
callback_demo.py`
from even import *
#中間函數(shù)
#接受一個(gè)生成偶數(shù)的函數(shù)作為參數(shù)
#返回一個(gè)奇數(shù)
def getOddNumber(k, getEvenNumber):
return 1 + getEvenNumber(k)
#起始函數(shù),這里是程序的主函數(shù)
def main():
k = 1
#當(dāng)需要生成一個(gè)2k+1形式的奇數(shù)時(shí)
i = getOddNumber(k, double)
print(i)
#當(dāng)需要一個(gè)4k+1形式的奇數(shù)時(shí)
i = getOddNumber(k, quadruple)
print(i)
#當(dāng)需要一個(gè)8k+1形式的奇數(shù)時(shí)
i = getOddNumber(k, lambda x: x * 8)
print(i)
if __name__ == "__main__":
main()異步處理有關(guān)的回調(diào)函數(shù)
def apply_ascyn(func, args, callback):
"""
func 函數(shù)的是處理的函數(shù)
args 表示的參數(shù)
callback 表示的函數(shù)處理完成后的 該執(zhí)行的動(dòng)作
"""
result = func(*args)
callback(result)
def add(x, y):
return x + y
def print_result(result):
print(result)
apply_ascyn(add, (2, 3), callback=print_result)這里print_result只能接收一個(gè)result的參數(shù),不能傳入其他信息。當(dāng)想讓回調(diào)函數(shù)訪問(wèn)其他變量或者特定環(huán)境的變量值的時(shí)候會(huì)遇到問(wèn)題。
使用一個(gè)綁定方法來(lái)代替這個(gè)簡(jiǎn)單函數(shù)。
def appy_async(func, args, *, callback):
result = func(*args)
#異步執(zhí)行的函數(shù) 同時(shí)將執(zhí)行后返回到這個(gè)早從這個(gè)函數(shù)跳出去
callback(result)
def add(x ,y):
return x + y
class ResultHandler(object):
def __init__(self):
self.sequence = 0
def handle(self, result):
self.sequence += 1
print("[{}] Got: {}".format(self.sequence, result))
resultHandler = ResultHandler()
appy_async(add, (2,3), callback=resultHandler.handle)
使用閉包代替上面的類來(lái)實(shí)現(xiàn)
def apply_async(func, args, *, callback):
result = func(*args)
callback(result)
def add(x ,y):
return x + y
def make_handler():
sequence = 0
def handler(result):
nonlocal sequence
sequence += 1
print("[{}] Got:{}".format(sequence, result))
return handler
handler = make_handler()
apply_async(add, (2,3), callback=handler)使用協(xié)程來(lái)完成異步操作
def apply_async(func, args, *, callback):
result = func(*args)
callback(result)
def add(x, y):
return x + y
def make_handler():
sequence = 0
while True:
result = yield
sequence += 1
print("[{}] Got:{}".format(sequence, result))
handle = make_handler()
next(handle)
apply_async(add, (2,3), callback=handle.send)博文參考:
python3回調(diào)函數(shù)(callback) - 知乎
總結(jié)
到此這篇關(guān)于Python中回調(diào)函數(shù)(callback)的文章就介紹到這了,更多相關(guān)Python 回調(diào)函數(shù)callback內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python代碼實(shí)現(xiàn)刪除一個(gè)list里面重復(fù)元素的方法
今天小編就為大家分享一篇關(guān)于Python代碼實(shí)現(xiàn)刪除一個(gè)list里面重復(fù)元素的方法,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-04-04
基于Python __dict__與dir()的區(qū)別詳解
下面小編就為大家?guī)?lái)一篇基于Python __dict__與dir()的區(qū)別詳解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
python之如何將標(biāo)簽轉(zhuǎn)化為one-hot(獨(dú)熱編碼)
這篇文章主要介紹了python之如何將標(biāo)簽轉(zhuǎn)化為one-hot(獨(dú)熱編碼)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06
python數(shù)據(jù)結(jié)構(gòu)leetcode338比特位計(jì)數(shù)算法
這篇文章主要介紹了力扣刷題中python數(shù)據(jù)結(jié)構(gòu)leetcode338比特位計(jì)數(shù)算法解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
Python使用Bokeh實(shí)現(xiàn)交互式圖表的創(chuàng)建
Bokeh?是一個(gè)流行的?Python?數(shù)據(jù)可視化庫(kù),可以生成高質(zhì)量的交互式圖表,這篇文章主要就介紹了Python如何使用Bokeh實(shí)現(xiàn)交互式圖表的創(chuàng)建,需要的可以參考一下2023-06-06
Python中線程threading.Thread的使用詳解
python的thread模塊是比較底層的模塊,python的threading模塊是對(duì)thread做了一些包裝的,可以更加方便的被使用。本文將為大家詳細(xì)介紹一下python中的線程threading.Thread()的使用,需要的可以參考一下2022-07-07

