python回調(diào)函數(shù)的使用方法
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)站建設(shè)、微信開發(fā)、成都小程序開發(fā)、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了隴西免費(fèi)建站歡迎大家使用!
在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡(jiǎn)稱回調(diào)(Callback),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計(jì)允許了底層代碼調(diào)用在高層定義的子程序
有兩種類型的回調(diào)函數(shù):
那么,在python中如何實(shí)現(xiàn)回調(diào)函數(shù)呢,看代碼:
代碼如下:
def my_callback(input):
print "function my_callback was called with %s input" % (input,)
def caller(input, func):
func(input)
for i in range(5):
caller(i, my_callback)
異步是計(jì)算機(jī)多線程的異步處理。與同步處理相對(duì),異步處理不用阻塞當(dāng)前線程來等待處理完成,而是允許后續(xù)操作,直至其它線程將處理完成,并回調(diào)通知此線程。
在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡(jiǎn)稱回調(diào)(Callback),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計(jì)允許了底層代碼調(diào)用在高層定義的子程序:
例如:
def?my_callback(input):
print?"function?my_callback?was?called?with?%s?input"?%?(input,)
def?caller(input,?func):
func(input)
for?i?in?range(5):
caller(i,?my_callback)
執(zhí)行結(jié)果是:
function?my_callback?was?called?with?0?input
function?my_callback?was?called?with?1?input
function?my_callback?was?called?with?2?input
function?my_callback?was?called?with?3?input
function?my_callback?was?called?with?4?input
1. 我忘了ctypes的操作方法是:
下面是從復(fù)制
因此,我們的回調(diào)函數(shù)接收整型指針,并且必須返回一個(gè)整數(shù)。首先,我們創(chuàng)建了回調(diào)函數(shù)的類型:
CMPFUNC = CFUNCTYPE(c_int, POINTER(c_int), POINTER(c_int))
對(duì)于優(yōu)先個(gè)回調(diào)函數(shù)中,我們簡(jiǎn)單地打印出我們得到,并返回0(;-):
def py_cmp_func(a, b):
print "py_cmp_func", a, b
return 0
創(chuàng)建C可調(diào)用的回調(diào)函數(shù):
cmp_func = CMPFUNC(py_cmp_func)
在使用ros::spin()的情況下,一般來說在初始化時(shí)已經(jīng)設(shè)置好所有消息的回調(diào),并且不需要其他背景程序運(yùn)行。這樣以來,每次消息到達(dá)時(shí)會(huì)執(zhí)行用戶的回調(diào)函數(shù)進(jìn)行操作,相當(dāng)于程序是消息事件驅(qū)動(dòng)的;而在使用ros::spinOnce()的情況下,一般來說僅僅使用回調(diào)不足以完成任務(wù),還需要其他輔助程序的執(zhí)行:比如定時(shí)任務(wù)、數(shù)據(jù)處理、用戶界面等。
關(guān)于消息接收回調(diào)機(jī)制在ROS官網(wǎng)上略有說明 (callbacks and spinning)??傮w來說其原理是這樣的:除了用戶的主程序以外,ROS的socket連接控制進(jìn)程會(huì)在后臺(tái)接收訂閱的消息,所有接收到的消息并不是立即處理,而是等到spin()或者spinOnce()執(zhí)行時(shí)才集中處理。所以為了保證消息可以正常接收,需要尤其注意spinOnce()函數(shù)的使用 (對(duì)于spin()來說則不涉及太多的人為因素)。
I. 對(duì)于速度較快的消息,需要注意合理控制消息隊(duì)列及spinOnce()的時(shí)間。例如,如果消息到達(dá)的頻率是100Hz,而spinOnce()的執(zhí)行頻率是10Hz,那么就要至少保證消息隊(duì)列中預(yù)留的大小大于10。
II. 如果對(duì)于用戶自己的周期性任務(wù),最好和spinOnce()并列調(diào)用。即使該任務(wù)是周期性的對(duì)于數(shù)據(jù)進(jìn)行處理,例如對(duì)接收到的IMU數(shù)據(jù)進(jìn)行Kalman濾波,也不建議直接放在回調(diào)函數(shù)中:因?yàn)榇嬖谕ㄐ沤邮盏牟淮_定性,不能保證該回調(diào)執(zhí)行在時(shí)間上的穩(wěn)定性。
// 示例代碼
ros::Rate r(100);
while (ros::ok())
{
libusb_handle_events_timeout(...); // Handle USB events
ros::spinOnce(); // Handle ROS events
r.sleep();
}
III. 最后說明一下將ROS集成到其他程序架構(gòu)時(shí)的情況。有些圖形處理程序會(huì)將main()包裹起來,此時(shí)就需要找到一個(gè)合理的位置調(diào)用ros::spinOnce()。比如對(duì)于OpenGL來說,其中有一個(gè)方法就是采用設(shè)置定時(shí)器定時(shí)調(diào)用的方法:
// 示例代碼
void timerCb(int value) {
ros::spinOnce();
}
glutTimerFunc(10, timerCb, 0);
glutMainLoop(); // Never returns