探究函数对象
1.引出问题
函数对象说白了就是类似于C语言中的函数指针,看下面的代码
我们知道调用一个函数的话,可以用C代码来实现
1
2
3
4
5
6
int sum(int a, int b)
{
return a + b;
}
int ret = sum(2, 1);
当然也可以这样用C++来实现
像这个operator重载了两个参数,叫做二元,重载一个叫做一元
1
2
3
4
5
6
7
8
9
10
11
class Sum
{
public:
int operator(int a, int b)
{
return a + b;
}
};
Sum sum;
int ret = sum(2, 1);
其中这个sum就叫做函数对象
把有operator()运算符重载函数的对象,称为函数对象或者称为仿函数
其中无论这个类或者结构体里是否还有其他的函数,但只要看有没有operator()运算符重载函数就行
函数对象一般来说只包含一个operator()运算符重载函数,但也可以包括其他的函数
在上面代码里
1
2
//int operator(int a, int b)
sum(2, 1) 等同于 sum.operator()(2, 1);
那既然他们两个都能实现,那用函数对象有啥好处吗
2.函数对象的好处
再看下面的例子
1
2
3
4
5
6
7
8
9
10
11
12
template<typename T>
bool compare(T a, T b)
{
return a > b;
}
int main()
{
cout << compare(10, 20) << endl;
cout << compare('b', 'y') << endl;
return 0;
}
现在如果想调用compare实现小于的话,会想,那把里面的a > b改成a < b不就好了,但万一情况是这个函数的实现是在库里怎么办,你总不能去改人家库里的东西吧,不现实
有人也会说,那我再实现一个这个不就好了
1
2
3
4
5
template<typename T>
bool compare(T a, T b)
{
return a < b;
}
那如果多了的话,你还一个个实现吗,也没啥必要
C语言的函数指针就可以解决这个问题
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
template<typename T>
bool greater(T a, T b)
{
return a > b;
}
template<typename T>
bool less(T a, T b)
{
return a < b;
}
// compare是C++的库函数模板
template<typename T>
bool compare(T a, T b)
{
return a > b;
}
int main()
{
cout << compare(10, 20) << endl;
cout << compare('b', 'y') << endl;
return 0;
}
因为compare是C++的库函数模板,所以他不能在函数里面写死,写成要么大于,要么小于的,所以应该再传入个参数,去接收个函数指针Compare
1
2
3
4
5
template<typename T, typenmae Compare>
bool compare(T a, T b, Compare comp)
{
return comp(a, b);//把ab当做这两个函数的参数传进来,根据传入的函数comp它的返回值我们再进行返回
}
这时候代码应该这样子
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
template<typename T>
bool mygreater(T a, T b)
{
return a > b;
}
template<typename T>
bool myless(T a, T b)
{
return a < b;
}
// compare是C++的库函数模板
template<typename T>
bool compare(T a, T b, Compare comp)
{
return comp(a, b);
}
int main()
{
cout << compare(10, 20, mygreater<int>) << endl;
cout << compare('b', 'y', myless<int>) << endl;
return 0;
}
compare(10, 20, greater<int>)
这个代码进去后首先进入compare函数里,用return comp(a, b),也就是return greater(10, 20),间接调用了greater函数
compare(T a, T b, Compare comp),第三个参数传入了greater的函数地址,也就是函数指针
但return comp(a, b) 通过函数指针调用函数,是没有办法内联的,效率很低,因为有函数的调用开销
因为内联是发生在编译阶段的,比如
1
2
3
4
5
6
7
8
inline void func() {}
template<typename T>
bool compare(T a, T b, Compare comp)
{
func()
return comp(a, b);
}
那函数在编译时候,走到func()时,就会自动找到inline void func() {},然后把这个函数展开,这样就省去了函数的调用开销了
但就算我们把刚才的代码改成这样的
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
template<typename T>
inline bool mygreater(T a, T b)
{
return a > b;
}
template<typename T>
inline bool myless(T a, T b)
{
return a < b;
}
// compare是C++的库函数模板
template<typename T>
bool compare(T a, T b, Compare comp)
{
return comp(a, b);
}
即使两个函数都加上inline,但编译时候走到return comp(a, b);时,编译器是不知道我调用的是 myless还是mygreater,完全不知道,除非我们明确写是调用了谁,只有在运行时候才会跑到这个地址上去找到对应的,因为函数指针只存储了函数地址,而没有任何函数体的信息
所以通过函数指针调用函数,是没有办法内联的,效率很低,因为有函数的调用开销
那我们再来看看C++函数对象的版本实现
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
template<typename T>
class mygreater
{
public:
bool operator()(T a, T b)
{
return a > b;
}
};
template<typename T>
class myless
{
public:
bool operator()(T a, T b)
{
return a < b;
}
};
// typename指向谁,谁就是类型名,比如这里的Compare就是一个类型
template<typename T, typename Compare>
bool compare(T a, T b, Compare comp)
{
return comp(a, b);
}
int main()
{
cout << compare(10, 20, mygreater<int>()) << endl;
cout << compare('b', 'y', myless<int>()) << endl;
}
//mygreater<int>是个类名,加个括号mygreater<int>()就是函数对象,因为是operator()重载
编译时在这里我们第一次调用compare函数的时候我们已经知道了他是调用哪个对象,比如compare(10, 20, mygreater<int>())是mygreater<int>,所以 comp 的类型就是 mygreater<int>。在第二次调用 compare 函数时,Compare 被指定为 myless<int> 类型,所以 comp 的类型就是 myless<int>。因此,在编译时,编译器就能确定 comp(a, b) 调用的是哪个函数对象,它可以在编译阶段将这个函数的函数体直接插入到调用它的地方。这个过程类似于内联函数的处理方式。
总的来说,就是函数对象相对于函数指针的一个优点是,它可以在编译时确定类型,这使得编译器能够更好地优化代码。由于函数对象是一个类,它可以重载 () 运算符,使得它可以像调用函数一样被调用。当我们通过函数对象调用一个函数时,编译器能够确定这个函数对象的类型,因此也能够确定它调用的是哪个函数。这样,编译器就可以将这个函数的函数体直接插入到调用它的地方,从而减少函数调用的开销。
3.修改其他的代码印证
再举个例子
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
#include <iostream>
#include <queue>
#include <set>
using namespace std;
int main()
{
priority_queue<int> que1; //vector 底层是大根堆
for(int i = 0; i < 10; i++)
{
que1.push(rand() % 100);
}
while(!que1.empty())
{
cout << que1.top() << " ";
que1.pop();
}
cout << endl;
}
这输出是从大到小的,那我们怎么从小到大呢,查看priority_queue源码我们会发现,第三个参数是默认less,我们是可以改的
如果我们改第三个的话,前两个参数也得手动写上
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <iostream>
#include <queue>
#include <set>
#include <vector>
using namespace std;
int main()
{
using MinHeap = priority_queue<int, vectpr<int>, greater<int>>; //小根堆
MinHeap que2;
for(int i = 0; i < 10; i++)
{
que2.push(rand() % 100);
}
while(!que2.empty())
{
cout << que2.top() << " ";
que2.pop();
}
cout << endl;
}
这样就是按从小到大排序了
同理set也可以这样玩