天天看點

【c++】 仿函數的概念詳解與基礎實作 四、仿函數

 四、仿函數

函數對象(仿函數)是一個類,不是一個函數。

函數對象(仿函數)重載了”() ”操作符使得它可以像函數一樣調用。

分類:假定某個類有一個重載的operator(),而且重載的operator()要求擷取一個參數,我們就将這個類稱為“一進制仿函數”(unary functor);相反,如果重載的operator()要求擷取兩個參數,就将這個類稱為“二進制仿函數”(binary functor)。

  • 第一種:先将“操作”設計為一個函數,再将函數指針當做算法的一個參數
  • 第二種:将“操作”設計為一個仿函數(在語言層面是一個class),再以該仿函數産生一個對象,并以此對象作為算法的一個參數

仿函數例一:比較大小

【c++】 仿函數的概念詳解與基礎實作 四、仿函數
  • 在上面的代碼中,第一種調用方式是使用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、模版函數對象使函數對象具有通用性,這也是它的優勢之一

繼續閱讀