C++11中增加了一些新的模板,今天我们简单介绍function模板。
头文件:<functional>
function是一个模板,当创建一个具体的function类型时,我们必须像其他类模板一样提供额外的信息。所谓的额外信息是指function类型能够表示的对象的调用形式。例如vector,我们在使用vector的时候必须要指定 vector中存放数据的类型,是int型、string型、还是double。我们必须明确指定。
function与函数指针比较相似,优点在于它允许用户在目标的实现上有更大的弹性,即目标既可以是普通函数,也可以是函数对象和类的成员函数。
一、普通函数:
首先我们给出一个普通的函数:
void foo(const string &s) { cout << s << endl; }
我们就可以在main函数中这样使用function
function<void(const string &s)> f =&foo; f("world");
测试结果为:world。上面function可以解释为:
声明并初始化了一个function,它返回的类型为void,并接受一个const string类型的参数,并且使 f 初始化为foo函数的地址。
二、类成员函数:
我们定义一个类,代码如下:
class Foo { public: void foo(int a) { cout << a << endl; } };
我们采用了以下一种调用方式:
//头文件 #include <iostream> #include <string> #include <functional> using namespace std;
//main函数 int main(int argc, const char *argv[]) { Foo f; //1 (mem_fun(&Foo::foo)(&f,123)); //2 function<void(int)> pf = bind(&Foo::foo, &f, placeholders::_1); pf(345); //3 function<void(Foo*, int)> pf2 =bind(&Foo::foo, std::placeholders::_1, std::placeholders::_2); pf2(&f,345); return 0; }
方式一:mem_fun,参见C++类的成员函数的指针和mem_fun适配器的用法
方式二:
function<void(int)> pf =bind(&Foo::foo, &f, placeholders::_1);
注意:
首先,bind 是一种 函数适配器 ,它可以改变参数的个数,顺序。
其次,bind中的参数依次为,类Foo的成员函数的地址、类Foo的一个对象f的地址,参数int 的占位符。
占位符_1、_2指的是实际函数调用实参的位置。而且,占位符必须是接连出现的,不可跳跃使用,bind 中的参数列表可以是 _1,_2,_3 .... 也可以是_1,_3,_2(即次序可以打乱),但是绝对不可以是这样的bind(&Foo::foo, _1, _3)
方式三:
function<void(Foo*, int)> pf2 =bind(&Foo::foo,std::placeholders::_1, std::placeholders::_2);
由于类的成员函数有个隐式参数this。所以,foo函数的实际参数有两个。
而,第三种方式把this这个隐式参数显示指出;故this对应bind中的占位符_1,int对应占位符_2.
三、实例:
#include <iostream> #include <string> #include <functional> using namespace std; using namespace std::placeholders; void test(int i, double d, const string &s) { cout << "i=" << i << " d=" << d << " s=" << s << endl; } int main(int argc, const char *argv[]) { function<void(int, double, const string &)>f1 = &test; f1(1, 3.14, "foo"); //1 //void(*)(int, double) function<void(double, int, const string &s)> f2 = bind(&test, _2, _1, _3); f2(4.14, 2, "hello"); //2 function<void(int, double) > f3= bind(&test, _1, _2, "world"); f3(3, 5.14); //3 function<void(const string&, int )> f4 = bind(&test, _2, 6.14, _1); f4("how", 4); //4 function<void(const string&, int, double)> f5 = bind(&test, _2, _3, _1); f5("are", 5, 7.14); //5 function<void(int)> f6 = bind(&test, _1, 8.14, "you"); f6(6); //6 function<void(const string &)> f7 =bind(&test, 7,9.14,_1); f7("thank"); //7 function<void()> f8 = bind(&test, 8, 10.14, "foobar"); f8(); return 0; }
时间: 2024-10-21 13:43:37