前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >探究函数对象

探究函数对象

作者头像
无敌清风蓝
发布2024-06-04 19:53:31
780
发布2024-06-04 19:53:31
举报
文章被收录于专栏:无敌清风蓝无敌清风蓝

1.引出问题

函数对象说白了就是类似于C语言中的函数指针,看下面的代码

我们知道调用一个函数的话,可以用C代码来实现

代码语言:javascript
复制
int sum(int a, int b)
{
    return a + b;
}

int ret = sum(2, 1);

当然也可以这样用C++来实现

像这个operator重载了两个参数,叫做二元,重载一个叫做一元

代码语言:javascript
复制
class Sum
{
public:
	int operator(int a, int b)
    {
        return a + b;
    }
};

Sum sum;
int ret = sum(2, 1);

其中这个sum就叫做函数对象

把有operator()运算符重载函数的对象,称为函数对象或者称为仿函数

其中无论这个类或者结构体里是否还有其他的函数,但只要看有没有operator()运算符重载函数就行

函数对象一般来说只包含一个operator()运算符重载函数,但也可以包括其他的函数

在上面代码里

代码语言:javascript
复制
//int operator(int a, int b)
sum(2, 1) 等同于 sum.operator()(2, 1);

那既然他们两个都能实现,那用函数对象有啥好处吗

2.函数对象的好处

再看下面的例子

代码语言:javascript
复制
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不就好了,但万一情况是这个函数的实现是在库里怎么办,你总不能去改人家库里的东西吧,不现实

有人也会说,那我再实现一个这个不就好了

代码语言:javascript
复制
template<typename T>
bool compare(T a, T b)
{
	return a < b;
}

那如果多了的话,你还一个个实现吗,也没啥必要

C语言的函数指针就可以解决这个问题

代码语言:javascript
复制
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

代码语言:javascript
复制
template<typename T, typenmae Compare>
bool compare(T a, T b, Compare comp)
{
	return comp(a, b);//把ab当做这两个函数的参数传进来,根据传入的函数comp它的返回值我们再进行返回
}

这时候代码应该这样子

代码语言:javascript
复制
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) 通过函数指针调用函数,是没有办法内联的,效率很低,因为有函数的调用开销

因为内联是发生在编译阶段的,比如

代码语言:javascript
复制
inline void func() {}

template<typename T>
bool compare(T a, T b, Compare comp)
{
    func()
	return comp(a, b);
}

那函数在编译时候,走到func()时,就会自动找到inline void func() {},然后把这个函数展开,这样就省去了函数的调用开销了

但就算我们把刚才的代码改成这样的

代码语言:javascript
复制
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++函数对象的版本实现

代码语言:javascript
复制
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.修改其他的代码印证

再举个例子

代码语言:javascript
复制
#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,我们是可以改的

如果我们改第三个的话,前两个参数也得手动写上

代码语言:javascript
复制
#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也可以这样玩

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023/06/17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1.引出问题
  • 2.函数对象的好处
  • 3.修改其他的代码印证
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档