關(guān)于c語(yǔ)言中回調(diào)函數(shù)的理解
前言
在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡(jiǎn)稱(chēng)回調(diào),是指通過(guò)函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計(jì)允許了底層代碼調(diào)用在高層定義的子程序。
這段話不是那么好理解,不同語(yǔ)言實(shí)現(xiàn)回調(diào)的方式有些許不同。其實(shí)可以這樣理解,回調(diào)就是在一個(gè)函數(shù)中調(diào)用另外一個(gè)函數(shù)。
在c語(yǔ)言中,回調(diào)是使用函數(shù)指針來(lái)實(shí)現(xiàn)的。 函數(shù)指針——顧名思義,是指向一個(gè)函數(shù)的指針。通常函數(shù)指針有兩個(gè)方面的用途,一個(gè)是轉(zhuǎn)換表(jump table),另一個(gè)是作為參數(shù)傳遞給一個(gè)函數(shù)。
下面是兩個(gè)函數(shù)指針的聲明
int(*f)(int,float);int*(*g[])(int,float);
前者把f聲明為一個(gè)函數(shù)指針,它所指的函數(shù)接受兩個(gè)參數(shù),分別是一個(gè)整型值和浮點(diǎn)型值,并返回一個(gè)整型值。
后者把g聲明為一個(gè)數(shù)組,數(shù)組的元素類(lèi)型是一個(gè)函數(shù)指針,它所指向的函數(shù)接受兩個(gè)參數(shù),分別是一個(gè)整型值和浮點(diǎn)型值,并返回一個(gè)整型指針。
需要注意的是,簡(jiǎn)單聲明一個(gè)函數(shù)指針并不意味著它馬上就可以使用。和其他指針一樣,對(duì)函數(shù)指針執(zhí)行間接訪問(wèn)之前必須把它初始化為指向某個(gè)函數(shù)。下面的代碼段說(shuō)明了一種初始化函數(shù)指針的方法。
intf(int);int(*pf)(int) = f;
第 2 個(gè)聲明創(chuàng)建了函數(shù)指針pf,并把它初始化為指向函數(shù)f。函數(shù)指針的初始化也可以通過(guò)一條賦值語(yǔ)句來(lái)完成。在函數(shù)指針的初始化之前具有f的原型是很重要的,否則編譯器就無(wú)法檢查f的類(lèi)型是否與pf所指向的類(lèi)型一致。
通過(guò)一個(gè)例子簡(jiǎn)單介紹回調(diào)函數(shù)的使用
大家應(yīng)該都對(duì)c語(yǔ)言的庫(kù)函數(shù)qsort有所了解,qsort聲明如下
void qsort(void*base,size_tnitems,size_tsize,int(*compar)(constvoid*,constvoid*))
可以看到,它的第三個(gè)參數(shù)是一個(gè)函數(shù)指針,傳入兩個(gè)沒(méi)有定義指針指向的類(lèi)型的參數(shù)a,b,返回一個(gè)整型值。實(shí)際上這里使用了回調(diào)函數(shù)。通過(guò)回調(diào)函數(shù),qsort可以在運(yùn)行時(shí)調(diào)用用戶定義的函數(shù)(底層代碼調(diào)用在高層定義的子程序)。
這里我們?cè)O(shè)計(jì)一個(gè)簡(jiǎn)單的sort函數(shù),來(lái)理解回調(diào)過(guò)程
1、定義函數(shù)指針
typedefint(*compar)(constint*a,constint*b);
2、自定義sort函數(shù),為了簡(jiǎn)單,這里使用冒泡排序
int*sort(int*nums,intn, compar cmp){int*target =malloc(n*sizeof(int));if(!target) perror("Memory error");memcpy(target, num, n *sizeof(int));for(inti =0; i < n; i++) {for(intj = i+1; j < n; j++) {if(cmp(target[i], target[j]) >0) {target[i] ^= target[j] ^= target[i] ^= target[j];}}}returntarget;}
3、實(shí)現(xiàn)函數(shù)回調(diào)
#include<stdio.h>#include<string.h>#include<stdlib.h>#include<errno.h>typedefint(*compar)(constint*a,constint*b);// 定義實(shí)現(xiàn)回調(diào)函數(shù)的"調(diào)用函數(shù)"int*sort(int*nums,intn, compar cmp){int*target =malloc(n*sizeof(int));if(!target) perror("Memory error");memcpy(target, num, n *sizeof(int));for(inti =0; i < n; i++) {for(intj = i+1; j < n; j++) {if(cmp(target[i], target[j]) <=0) {target[i] ^= target[j] ^= target[i] ^= target[j];}}}returntarget;}
// 定義回調(diào)函數(shù)intcmp1(inta,intb){returna < b;}intmain(intargc,charconst*argv[]){inta[10] = {1,4,3,1,10,4,5};int*x = bubble_sort(a,7, cmp1);for(inti =0; i <7; i++)printf("%d ", x[i]);printf("\n");return0;}
運(yùn)行結(jié)果:
1
1 1 3 4 4 5 10
調(diào)用函數(shù)向其函數(shù)中傳遞int (*compar)(const int *a, const int *b);這是int cmp1(int a, int b)函數(shù)的入口地址,即PC指針可以通過(guò)移動(dòng)到該地址執(zhí)行int cmp1(int a, int b)函數(shù),可以通過(guò)類(lèi)比數(shù)組來(lái)理解。
實(shí)現(xiàn)函數(shù)調(diào)用中,函數(shù)調(diào)用了“調(diào)用函數(shù)”,再在其中進(jìn)一步調(diào)用被“調(diào)用函數(shù)”。相比于主函數(shù)直接調(diào)用“被調(diào)函數(shù)”,這種方法為使用者,而不是開(kāi)發(fā)者提供了靈活的接口。另外,函數(shù)入口可以像變量一樣設(shè)定同樣為開(kāi)發(fā)者提供了靈活性。
總結(jié)
到此這篇關(guān)于c語(yǔ)言中回調(diào)函數(shù)理解的文章就介紹到這了,更多相關(guān)c語(yǔ)言中回調(diào)函數(shù)理解內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
實(shí)例解析C++設(shè)計(jì)模式編程中簡(jiǎn)單工廠模式的采用
這篇文章主要介紹了C++設(shè)計(jì)模式編程中簡(jiǎn)單工廠模式的采用實(shí)例,在簡(jiǎn)單工廠模式中程序往往利用封裝繼承來(lái)降低耦合度,需要的朋友可以參考下2016-03-03C語(yǔ)言中if語(yǔ)句加大括號(hào)和不加大括號(hào)的區(qū)別介紹
這篇文章主要給大家介紹了關(guān)于C語(yǔ)言中if語(yǔ)句加大括號(hào)和不加大括號(hào)的區(qū)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12Qt+QWidget實(shí)現(xiàn)簡(jiǎn)約美觀的加載動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Qt如何結(jié)合QWidget實(shí)現(xiàn)簡(jiǎn)約美觀的加載動(dòng)畫(huà),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-02-02C++實(shí)現(xiàn)打印1到最大的n位數(shù)
這篇文章主要介紹了C++實(shí)現(xiàn)打印1到最大的n位數(shù),并分析了實(shí)現(xiàn)代碼中語(yǔ)句的跳轉(zhuǎn)技巧,需要的朋友可以參考下2014-09-09Qt6.0開(kāi)發(fā)環(huán)境搭建步驟(圖文)
這篇文章主要介紹了Qt6.0開(kāi)發(fā)環(huán)境搭建步驟(圖文),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03C++ string與int的相互轉(zhuǎn)換(使用C++11)
本文主要介紹了C++ string與int的相互轉(zhuǎn)換(使用C++11),文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01