C/C++編程筆記:如何理解C語言中的回調(diào)函數(shù),零基礎(chǔ)也看得懂
在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),或簡稱回調(diào),是指通過函數(shù)參數(shù)傳遞到其它代碼的,某一塊可執(zhí)行代碼的引用。這一設(shè)計(jì)允許了底層代碼調(diào)用在高層定義的子程序。
這段話不是那么好理解,不同語言實(shí)現(xiàn)回調(diào)的方式有些許不同。其實(shí)可以這樣理解,回調(diào)就是在一個(gè)函數(shù)中調(diào)用另外一個(gè)函數(shù)。

在c語言中,回調(diào)是使用函數(shù)指針來實(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ù)組的元素類型是一個(gè)函數(shù)指針,它所指向的函數(shù)接受兩個(gè)參數(shù),分別是一個(gè)整型值和浮點(diǎn)型值,并返回一個(gè)整型指針。

需要注意的是,簡單聲明一個(gè)函數(shù)指針并不意味著它馬上就可以使用。和其他指針一樣,對函數(shù)指針執(zhí)行間接訪問之前必須把它初始化為指向某個(gè)函數(shù)。下面的代碼段說明了一種初始化函數(shù)指針的方法。
intf(int);int(*pf)(int) = f;
第 2 個(gè)聲明創(chuàng)建了函數(shù)指針pf,并把它初始化為指向函數(shù)f。函數(shù)指針的初始化也可以通過一條賦值語句來完成。在函數(shù)指針的初始化之前具有f的原型是很重要的,否則編譯器就無法檢查f的類型是否與pf所指向的類型一致。
通過一個(gè)例子簡單介紹回調(diào)函數(shù)的使用
大家應(yīng)該都對c語言的庫函數(shù)qsort有所了解,qsort聲明如下:
void qsort(void*base,size_tnitems,size_tsize,int(*compar)(constvoid*,constvoid*))
可以看到,它的第三個(gè)參數(shù)是一個(gè)函數(shù)指針,傳入兩個(gè)沒有定義指針指向的類型的參數(shù)a,b,返回一個(gè)整型值。實(shí)際上這里使用了回調(diào)函數(shù)。通過回調(diào)函數(shù),qsort可以在運(yùn)行時(shí)調(diào)用用戶定義的函數(shù)(底層代碼調(diào)用在高層定義的子程序)。

這里我們設(shè)計(jì)一個(gè)簡單的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、實(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指針可以通過移動到該地址執(zhí)行int cmp1(int a, int b)函數(shù),可以通過類比數(shù)組來理解。
實(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ù)”,這種方法為使用者,而不是開發(fā)者提供了靈活的接口。另外,函數(shù)入口可以像變量一樣設(shè)定同樣為開發(fā)者提供了靈活性。
另外如果你想更好的提升你的編程能力,學(xué)好C語言C++編程!彎道超車,快人一步!

另外,UP在主頁上傳了一些學(xué)習(xí)C/C++編程的視頻教程,有興趣或者正在學(xué)習(xí)的小伙伴一定要去看一看哦!會對你有幫助的~
分享(源碼、項(xiàng)目實(shí)戰(zhàn)視頻、項(xiàng)目筆記,基礎(chǔ)入門教程)
歡迎轉(zhuǎn)行和學(xué)習(xí)編程的伙伴,利用更多的資料學(xué)習(xí)成長比自己琢磨更快哦!
編程學(xué)習(xí)書籍分享:

編程學(xué)習(xí)視頻分享:
