1、前言
函數調用很好理解,即使剛學沒多久的朋友也知道函數調用是怎么實現的,即調用一個已經封裝好的函數,實現某個特定的功能。
把一個或者多個功能通過函數的方式封裝起來,對外只提供一個簡單的函數接口,然后在其他地方調用即可
2、函數調用方式
函數調用難道還能怎么調用?不就封裝好直接調用嗎???
函數調用方式分為兩種:直接調用和間接調用
直接調用
直接調用就是我們平常使用的方式,下面的方式就屬于直接調用了。
int SumFun(int a, int b)
{
return a + b;
}
int main()
{
// 直接調用定義好的函數
int sum = SumFun(5, 6);
printf("sum=%d", sum);
return 0;
}
間接調用
間接調用在初學時很難使用到,這是通過函數指針的方式實現的。
函數指針本質是一個指針變量,是一個指向函數的指針(函數本身也是有地址的,指向的是函數入口);
而指針函數本質是一個函數,其返回值為指針。
函數指針的用法如下:
typedef int (*FunctionCB)(int, int);
int SumFun(int a, int b)
{
return a + b;
}
int main()
{
// 將定義好的函數賦值給函數指針
FunctionCB pfnSum = SumFun;
// 通過函數指針間接調用
int sum = pfnSum(5, 6);
printf("sum=%d", sum);
return 0;
}
3、什么場景使用
函數指針在軟件架構分層設計中十分重要,因為分層設計中有一個設計原則,那就是下層函數不能直接調用上層函數,那么可以通過函數指針的方式實現;一般稱上層通過函數指針賦值給下層的函數為回調函數。
什么情況會存在需要下層程序需要調用上層程序的呢?
比如串口數據接收,雖然可以通過查詢的方式接收,但是遠不及通過串口中斷的方式接收及時,當接收完成時,需要立即通知上層讀取數據進行處理,而不是等待上層程序查詢讀取。
如何實現呢?
比如硬件抽象層/驅動層中的串口模塊實現函數
/************* UART.c 文件 ****************/
static UartRecvCB sg_pfnUartRecv;
// 設置數據幀接收處理回調函數
void UART_SetRecvCallback(UartRecvCB pfnUartRecv)
{
sg_pfnUartRecv = pfnUartRecv;
}
void UART_Task(void)
{
if (RecvEnd)
{
// 數據一幀接收完成立即調用
if (sg_pfnUartRecv != NULL)
{
sg_pfnUartRecv(UartRecvBuf, UartRecvLength);
}
}
}
/************* UART.h 文件 ****************/
typedef void (*UartRecvCB)(const char *, int);
extern void UART_SetRecvCallback(UartRecvCB pfnUartRecv);
extern void UART_Task(void);
應用層代碼中實現回調函數,并調用下層函數。
// 回調函數:串口數據處理
void OnUartRecvProcess(const char *pBuf, int length)
{
// 處理串口數據
printf("Recv: %s", pBuf);
}
int main()
{
UART_SetRecvCallback(OnUartRecvProcess);
while(1)
{
if (TimeFlag)
{
UART_Task();
}
}
}
上述示例中通過函數指針的方式間接調用了應用層的函數,而且并不違背分層設計原則。
如果看代碼不能立即理解的話,可以嘗試通過下圖理解:
-
接口
+關注
關注
33文章
8598瀏覽量
151163 -
封裝
+關注
關注
126文章
7901瀏覽量
142966 -
函數調用
+關注
關注
0文章
19瀏覽量
2585
發布評論請先 登錄
相關推薦
評論