C++中函式指標與函式物件

csu_zhengzy~發表於2018-11-13

今天學習c++中vector資料結構的底層實現,發現遍歷操作的實現之一用到了函式物件,花時間又複習了一下函式指標和函式物件。

函式指標:是指向函式的指標變數,在C編譯時,每一個函式都有一個入口地址,那麼這個指向這個函式的函式指標便指向這個地址。函式指標的用途是很大的,主要有兩個作用:用作呼叫函式和做函式的引數。

函式指標的宣告方法

資料型別標誌符 (指標變數名) (形參列表);一般函式的宣告為: 

        int func ( int x );

而一個函式指標的宣告方法為:

       int (*func) (int x);

前面的那個(*func)中括號是必要的,這會告訴編譯器我們宣告的是函式指標而不是宣告一個具有返回型為指標的函式,後面的形參要視這個函式指標所指向的函式形參而定。然而這樣宣告我們有時覺得非常繁瑣,於是typedef可以派上用場了,我們也可以這樣宣告:

        typedef int (*PF) (int x);

        PF pf;

這樣pf便是一個函式指標,方便了許多。當要使用函式指標來呼叫函式時,func(x)或者  (*fucn)(x) 就可以了,當然,函式指標也可以指向被過載的函式,編譯器會為我們區分這些過載的函式從而使函式指標指向正確的函式。
例子:

typedef void (*PFT) ( char ,int );
void bar(char ch, int i)
{
cout<<"bar "<<ch<<' '<<i<<endl;
return ;
}
PFT pft;
pft = bar;
pft('e',91);

例子中函式指標pft指向了一個已經宣告的函式bar(),然後通過pft來實現輸出字元和整型的目的。
函式指標另一個作用便是作為函式的引數,我們可以在一個函式的形參列表中傳入一個函式指標,然後便可以在這個函式中使用這個函式指標所指向的函式,這樣便可以使程式變得更加清晰和簡潔,而且這種用途技巧可以幫助我們解決很多棘手的問題,使用很小的代價就可獲得足夠大的利益(速度+複雜度)。

typedef void (*PFT) ( char ,int );
void bar(char ch, int i)
{
cout<<"bar "<<ch<<' '<<i<<endl;
return ;
}
void foo(char ch, int i, PFT pf)
{
pf(ch,i);
return ;
}
PFT pft;
pft = bar;
foo('e',12,pft);

前面是函式指標的應用,從一般的函式回撥意義上來說,函式物件和函式指標是相同的,但是函式物件卻具有許多函式指標不具有的有點,函式物件使程式設計更加靈活,而且能夠實現函式的內聯(inline)呼叫,使整個程式實現效能加速。
函式物件:這裡已經說明了這是一個物件,而且實際上只是這個物件具有的函式的某些功能,我們才稱之為函式物件,意義很貼切,如果一個物件具有了某個函式的功能,我們變可以稱之為函式物件。
如何使物件具有函式功能呢,很簡單,只需要為這個物件的操作符()進行過載就可以了,如下:

class A{
public:
int operator()(int x){return x;}
};
A a;
a(5);

這樣a就成為一個函式物件,當我們執行a(5)時,實際上就是利用了過載符號()。
函式物件既然是一個“類物件”,那麼我們當然可以在函式形參列表中呼叫它,它完全可以取代函式指標!如果說指標是C的標誌,類是C++特有的,那麼我們也可以說指標函式和函式物件之間的關係也是同前者一樣的!(雖然有些嚴密)。當我們想在形參列表中呼叫某個函式時,可以先宣告一個具有這種函式功能的函式物件,然後在形參中使用這個物件,他所作的功能和函式指標所作的功能是相同的,而且更加安全。
下面是一個例子:

class Func{
public:
int operator() (int a, int b)
{
cout<<a<<'+'<<b<<'='<<a+b<<endl;
return a;
}
};
int addFunc(int a, int b, Func& func)
{
func(a,b);
return a;
}
Func func;
addFunc(1,3,func);

上述例子中首先定義了一個函式物件類,並過載了()操作符,目的是使前兩個引數相加並輸出,然後在addFunc中的形參列表中使用這個類物件,從而實現兩數相加的功能。
如果運用泛型思維來考慮,可以定一個函式模板類,來實現一般型別的資料的相加:

 

class FuncT{
public:
template<typename T>
T operator() (T t1, T t2)
{
cout<<t1<<'+'<<t2<<'='<<t1+t2<<endl;
return t1;
}
};
template <typename T>
T addFuncT(T t1, T t2, FuncT& funct)
{
funct(t1,t2);
return t1;
}
FuncT funct;
addFuncT(2,4,funct);
addFuncT(1.4,2.3,funct);

大名鼎鼎的STL中便廣泛的運用了這項技術,詳細內容可參見候捷大師的一些泛型技術的書籍,不要以為函式物件的頻繁呼叫會使程式效能大大折扣,大量事實和實驗證明,正確使用函式物件的程式要比其他程式效能快很多!所以掌握並熟練運用函式物件才能為我們的程式加分,否則.......
如此看來,函式物件又為C++敞開了一道天窗,但隨之而來的便是一些複雜的問題和陷阱,如何去蔽揚利還需要我們不斷學習和探索

優勢:

1.函式物件可以有自己的狀態。我們可以在類中定義狀態變數,這樣一個函式物件在多次的呼叫中可以共享這個狀態; 函式物件。

2.有自己特有的型別。我們可以傳遞相應的型別作為引數來例項化相應的模板,比如說帶引數的函式形參。

相關文章