C&C++学习笔记(6)——function和bind

同样是C++11的新内容,用于函数封装。

本文仅供个人记录和复习,不用于其他用途

std::function

类模版std::function是一种通用、多态的函数封装。std::function的实例可以对任何可以调用的目标进行存储、复制、和调用操作,这些目标包括函数、lambda表达式、绑定表达式、以及其它函数对象等。

保存自由函数

1
2
3
4
5
6
7
8
void printA(int a)
{
cout << a << endl;
}
std::function<void (int a)> func;
func = printA;
func(2);

保存lambda表达式

1
2
std::function<void ()> func_1 = [](){cout << "hello world" << endl;};
func_1();

保存成员函数

1
2
3
4
5
6
7
8
9
10
struct Foo {
Foo(int num) : num_(num) {}
void print_add(int i) const { cout << num_+i << '\n'; }
int num_;
};
// 保存成员函数
std::function<void (const Foo&, int)> f_add_display = &Foo::print_add;
Foo foo(2);
f_add_display(foo, 1);

std::bind

bind是一组用于函数绑定的模板。在对某个函数进行绑定时,可以指定部分参数或全部参数,也可以不指定任何参数,还可以调整各个参数间的顺序。对于未指定的参数,可以使用占位符_1_2_3来表示。_1表示绑定后的函数的第1个参数,_2表示绑定后的函数的第2个参数,其他依次类推。
下面通过程序例子了解一下用法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#include <iostream>
using namespace std;
class A
{
public:
void fun_3(int k, int m)
{
cout << k << " " << m << endl;
}
};
void fun(int x,int y,int z)
{
cout << x << " " << y << " " << z << endl;
}
void fun_2(int &a, int &b)
{
a++;
b++;
cout << a << " " << b << endl;
}
int main(int argc, const char * argv[])
{
auto f1 = bind(fun, 1, 2, 3); // 表示绑定函数fun的第一,二,三个参数值为: 1 2 3
f1(); // print:1 2 3
auto f2 = bind(fun, placeholders::_1,placeholders::_2,3);
// 表示绑定函数 fun 的第三个参数为 3,而fun的第一,二个参数分别有调用 f2 的第一,二个参数指定
// print:1 2 3
f2(1,2);
auto f3 = bind(fun,placeholders::_2,placeholders::_1,3);
// 表示绑定函数 fun 的第三个参数为 3,而fun的第一,二个参数分别有调用 f3 的第二,一个参数指定
// 注意: f2 和 f3 的区别。
f3(1,2); // print:2 1 3
int n = 2;
int m = 3;
auto f4 = bind(fun_2, n,placeholders::_1);
f4(m); // print:3 4
cout << m << endl; // print:4 说明:bind对于不事先绑定的参数,通过std::placeholders传递的参数是通过引用传递的
cout << n << endl; // print:2 说明:bind对于预先绑定的函数参数是通过值传递的
A a;
auto f5 = bind(&A::fun_3, a, placeholders::_1, placeholders::_2);
f5(10,20); // print:10 20
std::function<void(int,int)> fc = std::bind(&A::fun_3, a, std::placeholders::_1, std::placeholders::_2);
fc(10,20); // print:10 20
return 0;
}