在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C語言回調函數,提升C技巧必備

Linux愛好者 ? 來源:Linux愛好者 ? 2023-01-16 14:28 ? 次閱讀

一、函數指針

在講回調函數之前,我們需要了解函數指針。

我們都知道,C語言的靈魂是指針,我們經常使用整型指針,字符串指針,結構體指針等。

int*p1;
char*p2;
STRUCT*p3;// STRUCT為我們定義的結構體

但是好像我們一般很少使用函數指針,我們一般使用函數都是直接使用函數調用。

下面我們來了解一下函數指針的概念和使用方法。

1. 概念

函數指針是指向函數的指針變量。

通常我們說的指針變量是指向一個整型、字符型或數組等變量,而函數指針是指向函數。

函數指針可以像一般函數一樣,用于調用函數、傳遞參數

函數指針的定義方式為:

函數返回值類型 (* 指針變量名) (函數參數列表);

“函數返回值類型”表示該指針變量可以指向具有什么返回值類型的函數;“函數參數列表”表示該指針變量可以指向具有什么參數列表的函數。這個參數列表中只需要寫函數的參數類型即可。

我們看到,函數指針的定義就是將“函數聲明”中的“函數名”改成“(指針變量名)”。但是這里需要注意的是:“(指針變量名)”兩端的括號不能省略,括號改變了運算符的優先級。如果省略了括號,就不是定義函數指針而是一個函數聲明了,即聲明了一個返回值類型為指針型的函數。

那么怎么判斷一個指針變量是指向變量的指針變量還是指向函數的指針變量呢?首先看變量名前面有沒有“”,如果有“”說明是指針變量;其次看變量名的后面有沒有帶有形參類型的圓括號,如果有就是指向函數的指針變量,即函數指針,如果沒有就是指向變量的指針變量。

最后需要注意的是,指向函數的指針變量沒有 ++ 和 – 運算。

一般為了方便使用,我們會選擇:

typedef 函數返回值類型 (* 指針變量名) (函數參數列表);

比如:

typedefint(*Fun1)(int);//聲明也可寫成int (*Fun1)(int x),但習慣上一般不這樣。
typedefint(*Fun2)(int,int);//參數為兩個整型,返回值為整型
typedefvoid(*Fun3)(void);//無參數和返回值
typedefvoid*(*Fun4)(void*);//參數和返回值都為void*指針

2. 如何用函數指針調用函數

給大家舉一個例子:

intFunc(intx);/*聲明一個函數*/
int(*p)(intx);/*定義一個函數指針*/
p=Func;/*將Func函數的首地址賦給指針變量p*/
p=&Func;/*將Func函數的首地址賦給指針變量p*/

賦值時函數 Func 不帶括號,也不帶參數。由于函數名 Func 代表函數的首地址,因此經過賦值以后,指針變量 p 就指向函數 Func() 代碼的首地址了。

下面來寫一個程序,看了這個程序你們就明白函數指針怎么使用了:

#include
intMax(int,int);//函數聲明
intmain(void)
{
int(*p)(int,int);//定義一個函數指針
inta,b,c;
p=Max;//把函數Max賦給指針變量p,使p指向Max函數
printf("pleaseenteraandb:");
scanf("%d%d",&a,&b);
c=(*p)(a,b);//通過函數指針調用Max函數
printf("a=%d
b=%d
max=%d
",a,b,c);
return0;
}
intMax(intx,inty)//定義Max函數
{
intz;
if(x>y)
{
z=x;
}
else
{
z=y;
}
returnz;
}

特別注意的是,因為函數名本身就可以表示該函數地址(指針),因此在獲取函數指針時,可以直接用函數名,也可以取函數的地址。

p=Max可以改成p=&Max
c=(*p)(a,b)可以改成c=p(a,b)

3. 函數指針作為某個函數的參數

既然函數指針變量是一個變量,當然也可以作為某個函數的參數來使用的。示例:

#include
#include

typedefvoid(*FunType)(int);
//前加一個typedef關鍵字,這樣就定義一個名為FunType函數指針類型,而不是一個FunType變量。
//形式同typedefint*PINT;
voidmyFun(intx);
voidhisFun(intx);
voidherFun(intx);
voidcallFun(FunTypefp,intx);
intmain()
{
callFun(myFun,100);//傳入函數指針常量,作為回調函數
callFun(hisFun,200);
callFun(herFun,300);

return0;
}

voidcallFun(FunTypefp,intx)
{
fp(x);//通過fp的指針執行傳遞進來的函數,注意fp所指的函數有一個參數
}

voidmyFun(intx)
{
printf("myFun:%d
",x);
}
voidhisFun(intx)
{
printf("hisFun:%d
",x);
}
voidherFun(intx)
{
printf("herFun:%d
",x);
}

輸出:

264e0410-955c-11ed-bfe3-dac502259ad0.jpg

4. 函數指針作為函數返回類型

有了上面的基礎,要寫出返回類型為函數指針的函數應該不難了,下面這個例子就是返回類型為函數指針的函數:

void(*func5(int,int,float))(int,int)
{
...
}

在這里,func5(int, int, float)為參數,其返回類型為void (*)(int, int)。在C語言中,變量或者函數的聲明也是一個大學問,想要了解更多關于聲明的話題,可以參考我之前的文章 - C專家編程》讀書筆記(1-3章)。這本書的第三章花了整整一章的內容來講解如何讀懂C語言的聲明。
5. 函數指針數組

在開始講解回調函數前,最后介紹一下函數指針數組。既然函數指針也是指針,那我們就可以用數組來存放函數指針。下面我們看一個函數指針數組的例子:

/*方法1*/
void(*func_array_1[5])(int,int,float);

/*方法2*/
typedefvoid(*p_func_array)(int,int,float);
p_func_arrayfunc_array_2[5];

上面兩種方法都可以用來定義函數指針數組,它們定義了一個元素個數為5,類型是 *void (*)(int, int, float) *的函數指針數組。

6. 函數指針總結

  1. 函數指針常量 :Max;函數指針變量:p;

  2. 數名調用如果都得如(*myFun)(10)這樣,那書寫與讀起來都是不方便和不習慣的。所以C語言的設計者們才會設計成又可允許myFun(10)這種形式地調用(這樣方便多了,并與數學中的函數形式一樣)。

  3. 在函數指針變量也可以存入一個數組內。數組的聲明方法:int (*fArray[10]) ( int );

二、回調函數

1. 什么是回調函數

我們先來看看百度百科是如何定義回調函數的:

回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用來調用其所指向的函數時,我們就說這是回調函數。回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用于對該事件或條件進行響應。
這段話比較長,也比較繞口。下面我通過一幅圖來說明什么是回調:

265c6ece-955c-11ed-bfe3-dac502259ad0.png

假設我們要使用一個排序函數來對數組進行排序,那么在主程序(Main program)中,我們先通過庫,選擇一個庫排序函數(Library function)。但排序算法有很多,有冒泡排序,選擇排序,快速排序,歸并排序。同時,我們也可能需要對特殊的對象進行排序,比如特定的結構體等。庫函數會根據我們的需要選擇一種排序算法,然后調用實現該算法的函數來完成排序工作。這個被調用的排序函數就是回調函數(Callback function)。

結合這幅圖和上面對回調函數的解釋,我們可以發現,要實現回調函數,最關鍵的一點就是要將函數的指針傳遞給一個函數(上圖中是庫函數),然后這個函數就可以通過這個指針來調用回調函數了。注意,回調函數并不是C語言特有的,幾乎任何語言都有回調函數。在C語言中,我們通過使用函數指針來實現回調函數。

我的理解是:把一段可執行的代碼像參數傳遞那樣傳給其他代碼,而這段代碼會在某個時刻被調用執行,這就叫做回調。

如果代碼立即被執行就稱為同步回調,如果過后再執行,則稱之為異步回調。

回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用來調用其所指向的函數時,我們就說這是回調函數。

回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用于對該事件或條件進行響應。

2. 為什么要用回調函數?

因為可以把調用者與被調用者分開,所以調用者不關心誰是被調用者。它只需知道存在一個具有特定原型和限制條件的被調用函數。

簡而言之,回調函數就是允許用戶把需要調用的方法的指針作為參數傳遞給一個函數,以便該函數在處理相似事件的時候可以靈活的使用不同的方法。

26698ed8-955c-11ed-bfe3-dac502259ad0.jpg

intCallback()// /
{
//TODO
return0;
}
intmain()// /
{
//TODO
Library(Callback);// /
//TODO
return0;
}

回調似乎只是函數間的調用,和普通函數調用沒啥區別。

但仔細看,可以發現兩者之間的一個關鍵的不同:在回調中,主程序把回調函數像參數一樣傳入庫函數。

這樣一來,只要我們改變傳進庫函數的參數,就可以實現不同的功能,這樣有沒有覺得很靈活?并且當庫函數很復雜或者不可見的時候利用回調函數就顯得十分優秀。

3. 怎么使用回調函數?

intCallback_1(inta)// /
{
printf("Hello,thisisCallback_1:a=%d",a);
return0;
}

intCallback_2(intb)// /
{
printf("Hello,thisisCallback_2:b=%d",b);
return0;
}

intCallback_3(intc)// /
{
printf("Hello,thisisCallback_3:c=%d",c);
return0;
}

intHandle(intx,int(*Callback)(int))// /
{
Callback(x);
}

intmain()
{
Handle(4,Callback_1);
Handle(5,Callback_2);
Handle(6,Callback_3);
return0;
}

如上述代碼:可以看到,Handle()函數里面的參數是一個指針,在main()函數里調用Handle()函數的時候,給它傳入了函數Callback_1()/Callback_2()/Callback_3()的函數名,這時候的函數名就是對應函數的指針,也就是說,回調函數其實就是函數指針的一種用法。

4. 下面是一個四則運算的簡單回調函數例子:

#include
#include

/****************************************
*函數指針結構體
***************************************/
typedefstruct_OP{
float(*p_add)(float,float);
float(*p_sub)(float,float);
float(*p_mul)(float,float);
float(*p_div)(float,float);
}OP;

/****************************************
*加減乘除函數
***************************************/
floatADD(floata,floatb)
{
returna+b;
}

floatSUB(floata,floatb)
{
returna-b;
}

floatMUL(floata,floatb)
{
returna*b;
}

floatDIV(floata,floatb)
{
returna/b;
}

/****************************************
*初始化函數指針
***************************************/
voidinit_op(OP*op)
{
op->p_add=ADD;
op->p_sub=SUB;
op->p_mul=&MUL;
op->p_div=÷
}

/****************************************
*庫函數
***************************************/
floatadd_sub_mul_div(floata,floatb,float(*op_func)(float,float))
{
return(*op_func)(a,b);
}

intmain(intargc,char*argv[])
{
OP*op=(OP*)malloc(sizeof(OP));
init_op(op);

/*直接使用函數指針調用函數*/
printf("ADD=%f,SUB=%f,MUL=%f,DIV=%f
",(op->p_add)(1.3,2.2),(*op->p_sub)(1.3,2.2),
(op->p_mul)(1.3,2.2),(*op->p_div)(1.3,2.2));

/*調用回調函數*/
printf("ADD=%f,SUB=%f,MUL=%f,DIV=%f
",
add_sub_mul_div(1.3,2.2,ADD),
add_sub_mul_div(1.3,2.2,SUB),
add_sub_mul_div(1.3,2.2,MUL),
add_sub_mul_div(1.3,2.2,DIV));

return0;
}

5. 回調函數實例(很有用)

一個GPRS模塊聯網的小項目,使用過的同學大概知道2G、4G、NB等模塊要想實現無線聯網功能都需要經歷模塊上電初始化、注冊網絡、查詢網絡信息質量、連接服務器等步驟,這里的的例子就是,利用一個狀態機函數(根據不同狀態依次調用不同實現方法的函數),通過回調函數的方式依次調用不同的函數,實現模塊聯網功能,如下:

/*********工作狀態處理*********/
typedefstruct
{
uint8_tmStatus;
uint8_t(*Funtion)(void);//函數指針的形式
}M26_WorkStatus_TypeDef;//M26的工作狀態集合調用函數


/**********************************************
**>M26工作狀態集合函數
***********************************************/
M26_WorkStatus_TypeDefM26_WorkStatus_Tab[]=
{
{GPRS_NETWORK_CLOSE,M26_PWRKEY_Off},//模塊關機
{GPRS_NETWORK_OPEN,M26_PWRKEY_On},//模塊開機
{GPRS_NETWORK_Start,M26_Work_Init},//管腳初始化
{GPRS_NETWORK_CONF,M26_NET_Config},//AT指令配置
{GPRS_NETWORK_LINK_CTC,M26_LINK_CTC},//連接調度中心
{GPRS_NETWORK_WAIT_CTC,M26_WAIT_CTC},//等待調度中心回復
{GPRS_NETWORK_LINK_FEM,M26_LINK_FEM},//連接前置機
{GPRS_NETWORK_WAIT_FEM,M26_WAIT_FEM},//等待前置機回復
{GPRS_NETWORK_COMM,M26_COMM},//正常工作
{GPRS_NETWORK_WAIT_Sig,M26_WAIT_Sig},//等待信號回復
{GPRS_NETWORK_GetSignal,M26_GetSignal},//獲取信號值
{GPRS_NETWORK_RESTART,M26_RESET},//模塊重啟
}
/**********************************************
**>M26模塊工作狀態機,依次調用里面的12個函數
***********************************************/
uint8_tM26_WorkStatus_Call(uint8_tStart)
{
uint8_ti=0;
for(i=0;i12;i++)
{
if(Start==M26_WorkStatus_Tab[i].mStatus)
{
returnM26_WorkStatus_Tab[i].Funtion();
}
}
return0;
}

所以,如果有人想做個NB模塊聯網項目,可以copy上面的框架,只需要修改回調函數內部的具體實現,或者增加、減少回調函數,就可以很簡潔快速的實現模塊聯網。

審核編輯 :李倩


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • C語言
    +關注

    關注

    180

    文章

    7605

    瀏覽量

    137000
  • 函數
    +關注

    關注

    3

    文章

    4333

    瀏覽量

    62696
  • 變量
    +關注

    關注

    0

    文章

    613

    瀏覽量

    28395

原文標題:C 語言回調函數,提升 C 技巧必備

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    同樣是函數,在CC++中有什么區別

    同樣是函數,在 CC++ 中有什么區別? 第一個返回值。 C語言函數可以不寫返回值類型,
    的頭像 發表于 11-29 10:25 ?334次閱讀

    使用C語言實現函數模板

      用C語言能不能實現一個通用的函數,既能完成整數的相加,又能完成浮點數的相加?
    的頭像 發表于 11-09 11:38 ?402次閱讀

    C語言中的socket編程基礎

    數據 步驟6:關閉socket 創建socket 在C語言中,創建socket需要使用socket()函數。這個函數需要兩個參數:域
    的頭像 發表于 11-01 16:51 ?346次閱讀

    定時器調函數能不能用ICACHE_FLASH_ATTR定義?

    非 OS SDK 在中斷處理函數中,請勿使用任何 ICACHE_FLASH_ATTR 定義的函數。 請問: 1、定時器和hw定時器的調函數
    發表于 07-22 06:33

    esp8266怎么找到調函數被調用的地方?

    esp8266里的程序怎么運行? 在user_init里注冊了espconn_regist_sentcb,espconn_regist_recvcb這幾個調函數,怎么找到這幾個
    發表于 07-10 08:24

    請問ESP32C3 I2S驅動支持調函數獲取數據嗎?

    目前I2S驅動只看到i2s_read接口獲取數據,但這是polling方式的。是否有注冊調函數,當數據到達是調用回頭函數通知上層app?
    發表于 06-28 06:11

    C語言內存泄漏問題原理

    內存泄漏問題只有在使用堆內存的時候才會出現,棧內存不存在內存泄漏問題,因為棧內存會自動分配和釋放。C語言代碼中堆內存的申請函數是malloc。
    發表于 03-19 11:38 ?531次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>內存泄漏問題原理

    調函數(callback)是什么?調函數的實現方法

    調函數是一種特殊的函數,它作為參數傳遞給另一個函數,并在被調用函數執行完畢后被調用。
    發表于 03-12 11:46 ?2977次閱讀

    函數指針與調函數的應用實例

    通常我們說的指針變量是指向一個整型、字符型或數組等變量,而函數指針是指向函數函數指針可以像一般函數一樣,用于調用函數、傳遞參數。
    的頭像 發表于 03-07 11:13 ?409次閱讀
    <b class='flag-5'>函數</b>指針與<b class='flag-5'>回</b><b class='flag-5'>調</b><b class='flag-5'>函數</b>的應用實例

    ??嵌入式中調函數的實現方法

    調函數的命名規范沒有固定的標準,但是根據通用慣例和編碼規范,調函數的命名應該能夠反映
    發表于 03-04 14:49 ?729次閱讀

    C語言中的可變參數介紹

    C 語言為這種情況提供了一個解決方案,它允許您定義一個函數,能根據具體的需求接受可變數量的參數
    發表于 02-28 14:00 ?323次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>中的可變參數介紹

    C語言中的動態內存管理講解

    本章將講解 C 中的動態內存管理。C 語言為內存的分配和管理提供了幾個函數。這些函數可以在 頭文件中找到。
    的頭像 發表于 02-23 14:03 ?399次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>中的動態內存管理講解

    如何使用C語言調函數降低程序耦合性呢?

    耦合性是程序結構中各個模塊之間相互關聯的度量。它取決于各個模塊之間接口的復雜程度、調用模塊的方式以及哪些信息通過接口。
    的頭像 發表于 02-22 11:34 ?618次閱讀
    如何使用<b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>回</b><b class='flag-5'>調</b><b class='flag-5'>函數</b>降低程序耦合性呢?

    c語言,c++,java,python區別

    C語言C++、Java和Python是四種常見的編程語言,各有優點和特點。 C語言
    的頭像 發表于 02-05 14:11 ?2431次閱讀

    如何解決C語言中的“訪問權限沖突”異常?C語言引發異常原因分析

    如何解決C語言中的“訪問權限沖突”異常?C語言引發異常原因分析? 在C語言中,訪問權限沖突異常通
    的頭像 發表于 01-12 16:03 ?5812次閱讀
    主站蜘蛛池模板: 国产视频分类| 天天躁夜夜| 伊人电影综合网| 三级a黄| 鲁丝一区二区三区| 钻石午夜影院| 无毒不卡| 国产高清一区二区三区四区| 黄 色 毛片免费| 国产三级精品在线观看| 四虎在线电影| 免费男女视频| 68日本xxxxxxx18| 97射射| 久久久久女人精品毛片九一| 澳门三级bd高清| 久久五月女厕所一区二区| 色免费在线| 欧美色视频日本| 黄色福利网| 夜色福利视频| 1024成人| 91精品久久国产青草| 午夜精品视频在线| 国外免费精品视频在线观看| 午夜免费啪啪| 日本免费一区二区老鸭窝| 国产综合在线视频| 日本19xxxxxxxxx69| 热re99久久精品国产99热| 69日本xxxxxxxxx29| 免费视频18| 色五月在线视频| 久久久久久88色偷偷| 亚洲系列_1页_mmyy11| 黄网在线看| 一级美女片| 轻点灬大ji巴太粗太长了啊h| 一区二区三区伦理| 欧美第一色| 影音先锋 色天使|