C++ 函數指針(初階)

来源:https://www.cnblogs.com/young520/archive/2022/09/13/16691042.html
-Advertisement-
Play Games

函數的地址是存儲其機器語言代碼的記憶體的開始地址。可以編寫將另一個函數的地址作為參數的函數,它允許在不同的時間傳遞不同函數的地址,這意味著可以在不同的時間使用不同的函數。 ...


函數的地址是存儲其機器語言代碼的記憶體的開始地址。可以編寫將另一個函數的地址作為參數的函數,它允許在不同的時間傳遞不同函數的地址,這意味著可以在不同的時間使用不同的函數。

1 函數指針類型

聲明指向函數的指針時,必須指定函數的返回類型以及函數的特征標(參數列表),可以首先編寫這種函數的原型,然後用 (*pf) 替換函數名,這樣 pf 就是這類函數的指針。以下麵的程式為例,要獲取函數的地址,只需使用函數名即可(後面不跟參數),這與數組地址有幾分相似,函數指針 pf 的類型是 double (*)(int),由於 pf 是指向 pam() 函數的指針,因此 (*pf) 是函數,使用函數指針調用函數時,C++ 將 pf(*pf) 看作是等價的(雖然前者是函數指針,後者是函數),將 pf() 用作函數調用與將 (*pf)() 用作函數調用,效果一樣。

//函數原型
double pam(int);

//聲明對應的函數指針
double (*pf)(int);

//賦值,也可在聲明時進行
pf = pam;

//使用函數指針調用函數,以下幾種方式等效
double x = pam(4);   //方式一
double x = (*pf)(4); //方式二
double x = pf(4);    //方式三

//輸出函數地址
cout << pam;  //值為0x001F1384
cout << pf;   //值為0x001F1384

對函數指針進行賦值時,對應函數的特征標和返回類型必須與 pf 相同,如果不相同,編譯器將拒絕這種賦值。例如函數指針類型 const double * (*)(const double *, int) 與下麵幾種函數匹配,它們的特征標看似不同,但實際上相同,還可使用 auto 關鍵字自動推斷函數指針的類型。

//函數原型
const double * f1(const double ar[], int n);
const double * f2(const double * ar, int n);
const double * f3(const double [], int);
const double * f4(const double *, int);

//聲明函數指針
const double * (*pf)(const double *, int);

//賦值
pf = f1;
pf = f2;
pf = f3;
pf = f4;

//可使用自動類型推斷
auto pff = f1;

假設要設計一個名為 estimate() 的函數,用於估算編寫指定行數的代碼所需的時間,而且允許每個程式員提供自己的演算法來估算時間,此時可將函數地址作為該函數的輸入參數,其函數原型可使用如下聲明,需保證每個程式員提供的函數特征標和返回類型都一致且與 double (*)(int) 匹配。

//將函數指針作為函數參數
void estimate(int lines, double (*pf)(int));

2 函數指針數組

指向函數指針數組的指針通常用於類的虛方法實現中,細節通常由編譯器自動處理。如下程式所示,函數指針數組大體性質與一維數組相似,其中需要註意的是:

  • 運算符 ()[] 的優先順序比 * 要高,因此需在合適的地方使用括弧 () 提高 * 的優先順序。
  • 無法將指針算術運用於函數名,即出現 fb+1arrpf[i]+1 時編譯器會報錯。
  • 無法將 sizeof() 運用於函數名 fb ,但可用於 arrpf[i],即出現 sizeof(fb) 時編譯器會報錯,但 sizeof(arrpf[i]) 則不會。
//函數原型
double fa(int);
double fb(int);
double fc(int);
double fd(int);

//聲明並初始化函數指針數組
double (*arrpf[4])(int) = {fa,fb,fc,fd};

//聲明並初始化指向函數指針數組第一個元素的指針,以下三種方式對arrpfb等效
double (**arrpfb)(int) = arrpf;     //方式一
double (**arrpfb)(int) = &arrpf[0]; //方式二
auto arrpfb = &arrpf[0];            //方式三

//聲明並初始化指向整個函數指針數組的指針,以下兩種方式對arrpfc等效
double (*(*arrpfc)[4])(int) = &arrpf; //方式一
auto arrpfc = &arrpf;                 //方式二

//調用函數fb,以下幾種方式等效
int x = 5;
double y = fb(x);

double y = arrpf[1](x);
double y = (*arrpf[1])(x);
double y = (*(arrpf+1))(x);
double y = (**(arrpf+1))(x);

double y = arrpfb[1](x);
double y = (*arrpfb[1])(x);
double y = (*(arrpfb+1))(x);
double y = (**(arrpfb+1))(x);

double y = (*arrpfc)[1](x);
double y = (*(*arrpfc)[1])(x);
double y = (*(*arrpfc+1))(x);
double y = (**(*arrpfc+1))(x);

//應用指針算術時單位1表示的位元組數(32系統)
cout << int(arrpf+1)-int(arrpf);        //結果為4
cout << int(&arrpf[0]+1)-int(&arrpf[0]);//結果為4
cout << int(&arrpf+1)-int(&arrpf);      //結果為16

cout << int(arrpfb+1)-int(arrpfb);      //結果為4
cout << int(arrpfc+1)-int(arrpfc);      //結果為16

//應用sizeof()獲得記憶體量大小(32系統)
cout << sizeof(arrpf);    //結果為16
cout << sizeof(&arrpf[0]);//結果為4
cout << sizeof(&arrpf);   //結果為4
cout << sizeof(arrpf[0]); //結果為4

cout << sizeof(arrpfb);   //結果為4
cout << sizeof(arrpfc);   //結果為4

3 使用 typedef 進行簡化

可將 typedef 關鍵字用於函數指針類型,簡化程式的編寫,如下所示,此時 p_fun 就是函數指針類型 double (*)(int) 的別名,上述數組以及指針聲明可採用以下等效形式。

//指定函數指針別名
typedef double (*p_fun)(int);

//聲明並初始化函數指針數組,與前面等效
p_fun arrpf[4] = {fa,fb,fc,fd};

//聲明並初始化指向函數指針數組第一個元素的指針,與前面等效
p_fun* arrpfb = &arrpf[0]; 

//聲明並初始化指向整個函數指針數組的指針,與前面等效
p_fun (*arrpfc)[4] = &arrpf;

本文作者:木三百川

本文鏈接:https://www.cnblogs.com/young520/p/16691042.html

版權聲明:本文系博主原創文章,著作權歸作者所有。商業轉載請聯繫作者獲得授權,非商業轉載請附上出處鏈接。遵循 署名-非商業性使用-相同方式共用 4.0 國際版 (CC BY-NC-SA 4.0) 版權協議。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 由於 vite 出現的時間不是很久,基於 vite 創建的項目沒有 vue-cli 那麼完整,如果要使用 vue 全家桶、ESLint 等,還需要開發人員手動添加和配置,步驟稍多,略繁瑣。雖然在創建項目時可以選擇 *Customize with create-vue*,但我由於網路問題,一直沒有成功... ...
  • 每日3題 34 以下代碼執行後,控制臺中的輸出內容為? const num = { a: 10, add() { return this.a + 2; }, reduce: () => this.a - 2, }; console.log(num.add()); console.log(num.re ...
  • Electron 不錯,但也不是完美的。 Electron 帶來了很多優秀的桌面軟體,但並不一定總是適合我們的需求。 多個選擇總是好事! ▶ 我使用 Electron 遇到的一些麻煩 1、Electron 太大了! 2、每一個 Electron 寫的軟體都要重覆地帶一個 Electron …… 升級 ...
  • #【設計模式】工廠模式 相對來說,寫的比較亂,但是看一下實例,其實理解很快 ##抽象工廠模式(這裡主要介紹抽象工廠模式) 核心的工廠類不再負責所有對象的創建,而是將具體的創建工作交給子類去做。這個類則搖身一變變成了一個抽象工廠角色,僅僅負責給出具體工廠子類必須實現的介面。 舉個例子: 我有個雞廠,我 ...
  • Hello,我是你們的好朋友小烤鴨,這過了個中秋節,胡吃海喝了兩日,學習拉下了,今天返崗,繼續把我們的設計模式撿起,希望我能堅持完這個系列吧,下麵我們就進入正題吧。 在軟體開發過程中,我們需要重覆使用某個對象的時候,如果重覆地new這個對象,不停地申請記憶體空間,會造成記憶體空間的極大浪費,在之後程式運 ...
  • 本文主要給出了電商平臺通用取貨碼的完整設計思路,按此思路可以實現大多數核銷類券碼的生成問題,也能夠滿足SaaS化的業務 ...
  • 我的博客在看到這個標題時候肯定有人會想,我寫SQL直接在資料庫工具上執行就行了啊,工具會自動識別註釋的,就是不用工具,把SQL寫到存儲過程里,資料庫也會識別註釋不執行的,幹嘛非要去掉,費力不討好。 其實是最近在做一個項目,需要在行雲庫里執行SQL,並且SQL是寫在腳本上的,通過JDBC調用,眾所周知 ...
  • 前言 CMake是C++的必學部分,本篇文章從安裝環境開始,通過使用CMake構建一個最簡單的cpp項目和g++直接編譯作對比瞭解CMake的構建過程,為接下來深入學習CMake打下基礎。 一、系統環境 操作系統:Windows7 sp1 專業版 構建工具:CMake 3.24.1 下載地址:htt ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...