關(guān)于c語言中回調(diào)函數(shù)的理解
前言
在計算機程序設(shè)計中,回調(diào)函數(shù),或簡稱回調(diào),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計允許了底層代碼調(diào)用在高層定義的子程序。
這段話不是那么好理解,不同語言實現(xiàn)回調(diào)的方式有些許不同。其實可以這樣理解,回調(diào)就是在一個函數(shù)中調(diào)用另外一個函數(shù)。
在c語言中,回調(diào)是使用函數(shù)指針來實現(xiàn)的。 函數(shù)指針——顧名思義,是指向一個函數(shù)的指針。通常函數(shù)指針有兩個方面的用途,一個是轉(zhuǎn)換表(jump table),另一個是作為參數(shù)傳遞給一個函數(shù)。
下面是兩個函數(shù)指針的聲明
int(*f)(int,float);int*(*g[])(int,float);
前者把f聲明為一個函數(shù)指針,它所指的函數(shù)接受兩個參數(shù),分別是一個整型值和浮點型值,并返回一個整型值。
后者把g聲明為一個數(shù)組,數(shù)組的元素類型是一個函數(shù)指針,它所指向的函數(shù)接受兩個參數(shù),分別是一個整型值和浮點型值,并返回一個整型指針。
需要注意的是,簡單聲明一個函數(shù)指針并不意味著它馬上就可以使用。和其他指針一樣,對函數(shù)指針執(zhí)行間接訪問之前必須把它初始化為指向某個函數(shù)。下面的代碼段說明了一種初始化函數(shù)指針的方法。
intf(int);int(*pf)(int) = f;
第 2 個聲明創(chuàng)建了函數(shù)指針pf,并把它初始化為指向函數(shù)f。函數(shù)指針的初始化也可以通過一條賦值語句來完成。在函數(shù)指針的初始化之前具有f的原型是很重要的,否則編譯器就無法檢查f的類型是否與pf所指向的類型一致。
通過一個例子簡單介紹回調(diào)函數(shù)的使用
大家應(yīng)該都對c語言的庫函數(shù)qsort有所了解,qsort聲明如下
void qsort(void*base,size_tnitems,size_tsize,int(*compar)(constvoid*,constvoid*))
可以看到,它的第三個參數(shù)是一個函數(shù)指針,傳入兩個沒有定義指針指向的類型的參數(shù)a,b,返回一個整型值。實際上這里使用了回調(diào)函數(shù)。通過回調(diào)函數(shù),qsort可以在運行時調(diào)用用戶定義的函數(shù)(底層代碼調(diào)用在高層定義的子程序)。
這里我們設(shè)計一個簡單的sort函數(shù),來理解回調(diào)過程
1、定義函數(shù)指針
typedefint(*compar)(constint*a,constint*b);
2、自定義sort函數(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;}
3、實現(xiàn)函數(shù)回調(diào)
#include<stdio.h>#include<string.h>#include<stdlib.h>#include<errno.h>typedefint(*compar)(constint*a,constint*b);// 定義實現(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;}
運行結(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指針可以通過移動到該地址執(zhí)行int cmp1(int a, int b)函數(shù),可以通過類比數(shù)組來理解。
實現(xiàn)函數(shù)調(diào)用中,函數(shù)調(diào)用了“調(diào)用函數(shù)”,再在其中進一步調(diào)用被“調(diào)用函數(shù)”。相比于主函數(shù)直接調(diào)用“被調(diào)函數(shù)”,這種方法為使用者,而不是開發(fā)者提供了靈活的接口。另外,函數(shù)入口可以像變量一樣設(shè)定同樣為開發(fā)者提供了靈活性。
總結(jié)
到此這篇關(guān)于c語言中回調(diào)函數(shù)理解的文章就介紹到這了,更多相關(guān)c語言中回調(diào)函數(shù)理解內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ string與int的相互轉(zhuǎn)換(使用C++11)
本文主要介紹了C++ string與int的相互轉(zhuǎn)換(使用C++11),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01