四、仿函數
函數對象(仿函數)是一個類,不是一個函數。
函數對象(仿函數)重載了”() ”操作符使得它可以像函數一樣調用。
分類:假定某個類有一個重載的operator(),而且重載的operator()要求擷取一個參數,我們就将這個類稱為“一進制仿函數”(unary functor);相反,如果重載的operator()要求擷取兩個參數,就将這個類稱為“二進制仿函數”(binary functor)。
- 第一種:先将“操作”設計為一個函數,再将函數指針當做算法的一個參數
- 第二種:将“操作”設計為一個仿函數(在語言層面是一個class),再以該仿函數産生一個對象,并以此對象作為算法的一個參數
仿函數例一:比較大小
- 在上面的代碼中,第一種調用方式是使用comp的定義的一個對象,然後通過這個對象來調用操作符(),來實作兩個數組的比較的;
- 對于第二個調用comp()(1, 2)是産生一個臨時(無名的)對象。
函數對象的作用:
STL提供的算法往往都有兩個版本,其中一個版本表現出最常用的某種運算,另一版本則允許使用者通過template參數的形式來指定所要采取的政策。
//函數對象是重載了函數調用符号的類
class MyPrint
{
public:
MyPrint()
{
m_Num = 0;
}
int m_Num;
public:
void operator() (int num)
{
cout << num << endl;
m_Num++;
}
};
//函數對象
//重載了()操作符的類執行個體化的對象,可以像普通函數那樣調用,可以有參數 ,可以有傳回值
void test01()
{
MyPrint myPrint;
myPrint(20);
}
// 函數對象超出了普通函數的概念,可以儲存函數的調用狀态
void test02()
{
MyPrint myPrint;
myPrint(20);
myPrint(20);
myPrint(20);
cout << myPrint.m_Num << endl;
}
void doBusiness(MyPrint print,int num)
{
print(num);
}
//函數對象作為參數
void test03()
{
//參數1:匿名函數對象
doBusiness(MyPrint(),30);
}
總結:
1、函數對象通常不定義構造函數和析構函數,是以在構造和析構時不會發生任何問題,避免了函數調用的運作時問題。
2、函數對象超出普通函數的概念,函數對象可以有自己的狀态
3、函數對象可内聯編譯,性能好。用函數指針幾乎不可能
4、模版函數對象使函數對象具有通用性,這也是它的優勢之一