C++——function和bind的一些应用

function是一种类模板,重载了operator()函数调用操作符,所以每一个function类的对象都是一个函数对象。

我们可以这样使用function模板:

#include <iostream>
#include <string>
#include <vector>
#include <functional>
using namespace std;

void foo(const string &s)
{
    cout << s << endl;
}


int main(int argc, const char *argv[])
{
    void (*pFunc) (const string &) = &foo;
    pFunc("bar");

    function<void (const string&)> f = &foo;
    f("bar");
    

    return 0;
}

这样f()函数与函数指针pFunc()的作用是一样的,我们可以通过传递对应的参数来调用函数。

 

在使用function模板时,我们可以通过调用bind()函数来改变参数的个数、顺序等。bind()是一种函数适配器。

来看一个简单的示例:

 1 #include <iostream>
 2 #include <string>
 3 #include <vector>
 4 #include <functional>
 5 using namespace std;
 6 
 7 class Foo
 8 {
 9     public:
10         void foo(int i) { cout << i << endl; }        
11 
12         static void bar(double d) { cout << d << endl; }
13 
14 };
15 
16 int main(int argc, const char *argv[])
17 {
18     //mem_fun void (*)(Foo *, int)
19     Foo f;
20     (mem_fun(&Foo::foo))(&f, 123); 
21 
22     function<void (int)> pf = bind(&Foo::foo, 
23                                    &f, 
24                                    std::placeholders::_1);
25     pf(345);
26 
27     function<void (Foo*, int)> pf2 = bind(&Foo::foo,
28                                           std::placeholders::_1,
29                                           std::placeholders::_2);
30 
31     pf2(&f, 456);
32 
33     function<void (int, Foo*)> pf3 = bind(&Foo::foo,
34                     std::placeholders::_2,
35                     std::placeholders::_1);
36 
37     pf3(567, &f);
38 
39 
40 
41     return 0;
42 }

 

在bind()函数中,我们首先传递要绑定的函数地址,然后根据我们重新定义的函数即function<void (Foo*, int)> pF2来确定我们的参数个数及顺序。

void foo(int i);
function<void (Foo*, int)> pf2 = bind(&Foo::foo,
                                          std::placeholders::_1,
                                          std::placeholders::_2);

 

由于foo()函数是类成员函数,它有一个隐式参数,所以我们原函数含有两个参数。根据function模板,我们需要按照Foo *, int的顺序传递参数,在原函数中,
Foo *是第一个参数,对应放置的位置为1, int是第二个参数, 对应的位置为2, 这样,我们调用std::palceholders::来将参数的位置确定下来。

占位符_1_2指的是实际函数调用实参的位置

下面是一个有三个参数的函数对应的各种function可能,我们可以从中自己体会,也可以自己尝试重新实现以下。

代码如下:

 1 #include <iostream>
 2 #include <string>
 3 #include <vector>
 4 #include <functional>
 5 using namespace std;
 6 
 7 void test(int i, double d, const string &s)
 8 {
 9     cout << "i = " << i << " d = " << d << " s = " << s << endl;
10 }
11 int main(int argc, const char *argv[])
12 {
13     function<void (int, double, const string&)> f1 = &test;
14     f1(12, 3.14, "foo");
15 
16     //1.void (*)(int, double)
17     function<void (int, double)> f2 =
18     std::bind(&test, std::placeholders::_1, std::placeholders::_2, "foo");
19     f2(12, 3.14);
20 
21     //2.void (*)(double, int, const string &)
22     function<void (double, int, const string &)> f3 =
23     std::bind(&test, std::placeholders::_2, std::placeholders::_1, std::placeholders::_3);
24     f3(3.14, 12, "foo");
25 
26     //3.void (*)(const string &, int)
27     function<void (const string &, int)> f4 =
28     std::bind(&test, std::placeholders::_2, 3.14, std::placeholders::_1);
29     f4("foo", 12);
30 
31     //4. void (*) (const string &, int, double)
32     function<void (const string &, int, double)> f5 =
33     std::bind(&test, std::placeholders::_2, std::placeholders::_3, std::placeholders::_1);
34     f5("foo", 12, 3.14);
35 
36     //5. void (*)(int)
37     function<void (int)> f6 =
38     std::bind(&test, std::placeholders::_1, 3.14, "foo");
39     f6(12);
40 
41     //6 void(*)(const string &)
42     function<void (const string &)> f7 =
43     std::bind(&test, 12, 3.14, std::placeholders::_1);
44     f7("foo");
45 
46     //7. void (*)()
47     function<void ()> f8 =
48     std::bind(&test, 12, 3.14, "foo");
49     f8();
50 }
View Code

 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。