網站首頁 編程語言 正文
函數指針
概念
指針是一個變量,是用來指向內存地址的。一個程序運行時,所有和運行相關的物件都是需要加載到內存中,這就決定了程序運行時的任何物件都可以用指針來指向它。函數是存放在內存代碼區域內的,它們同樣有地址,因此同樣可以用指針來存取函數,把這種指向函數入口地址的指針稱為函數指針。
先來看一個Hello World程序
int main(int argc,char* argv[]) { printf("Hello World!\n"); return 0; }
然后,采用函數調用的形式來實現
void Invoke(char* s); int main(int argc,char* argv[]) { Invoke("Hello World!\n"); return 0; } void Invoke(char* s) { printf(s); }
用函數指針的方式來實現
void Invoke(char* s); int main() { void (*fp)(char* s); //聲明一個函數指針(fp) fp=Invoke; //將Invoke函數的入口地址賦值給fp fp("Hello World!\n"); //函數指針fp實現函數調用 return 0; } void Invoke(char* s) { printf(s); }
由上知道:函數指針函數的聲明之間唯一區別就是,用指針名(*fp)代替了函數名Invoke,這樣這聲明了一個函數指針,然后進行賦值fp=Invoke就可以進行函數指針的調用了。聲明函數指針時,只要函數返回值類型、參數個數、參數類型等保持一致,就可以聲明一個函數指針了。注意,函數指針必須用括號括起來 void (*fp)(char* s)。
實際中,為了方便,通常用宏定義的方式來聲明函數指針,實現程序如下:
typedef void (*FP)(char* s); void Invoke(char* s); int main(int argc,char* argv[]) { FP fp; //通常是用宏FP來聲明一個函數指針fp fp=Invoke; fp("Hello World!\n"); return 0; } void Invoke(char* s) { printf(s); }
函數指針數組
下面用程序對函數指針數組來個大致了解:
#include <iostream> #include <string> using namespace std; typedef void (*FP)(char* s); void f1(char* s){cout<<s;} void f2(char* s){cout<<s;} void f3(char* s){cout<<s;} int main(int argc,char* argv[]) { void* a[]={f1,f2,f3}; //定義了指針數組,這里a是一個普通指針 a[0]("Hello World!\n"); //編譯錯誤,指針數組不能用下標的方式來調用函數 FP f[]={f1,f2,f3}; //定義一個函數指針的數組,這里的f是一個函數指針 f[0]("Hello World!\n"); //正確,函數指針的數組進行下標操作可以進行函數的間接調用 return 0; }
回調函數
概念
回調函數,顧名思義,就是使用者自己定義一個函數,使用者自己實現這個函數的程序內容,然后把這個函數作為參數傳入別人(或系統)的函數中,由別人(或系統)的函數在運行時來調用的函數。函數是你實現的,但由別人(或系統)的函數在運行時通過參數傳遞的方式調用,這就是所謂的回調函數。簡單來說,就是由別人的函數運行期間來回調你實現的函數。
標準Hello World程序
int main(int argc,char* argv[]) { printf("Hello World!\n"); return 0; }
將它修改成函數回調樣式
//定義回調函數 void PrintfText() { printf("Hello World!\n"); } //定義實現回調函數的"調用函數" void CallPrintfText(void (*callfuct)()) { callfuct(); } //在main函數中實現函數回調 int main(int argc,char* argv[]) { CallPrintfText(PrintfText); return 0; }
修改成帶參的回調樣式
//定義帶參回調函數 void PrintfText(char* s) { printf(s); } //定義實現帶參回調函數的"調用函數" void CallPrintfText(void (*callfuct)(char*),char* s) { callfuct(s); } //在main函數中實現帶參的函數回調 int main(int argc,char* argv[]) { CallPrintfText(PrintfText,"Hello World!\n"); return 0; }
總結
回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用為調用它所指向的函數時,我們就說這是回調函數。回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用于對該事件或條件進行響應。
原文鏈接:https://blog.csdn.net/weixin_45713725/article/details/122545764
相關推薦
- 2023-02-15 C#?9使用foreach擴展的示例詳解_C#教程
- 2022-03-14 Android 截屏在surfaceview上失敗的問題
- 2023-10-16 element--el-input限制輸入為數字且必須大于0
- 2022-05-12 python2中input()漏洞
- 2022-12-28 Android?ViewPager2?+?Fragment?聯動效果的實現思路_Android
- 2022-04-15 python3?cmp實現方式_python
- 2022-09-15 Android?Jetpack庫剖析之Lifecycle組件篇_Android
- 2022-10-17 Python?pywin32實現word與Excel的處理_python
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支