上篇博客在为大家介绍lambda是先由find_if函数引入问题,由于find_if函数的第三个参数所要求的可调用对象只能由一个谓词,所以我们最后选择的解决方法是用lambda表达式。那么此问题还有其他的解决方法么?本篇博客我就为大家带来新的解决方法bind函数
1.bind函数的基本介绍
bind函数的最根本的作用就是可以把一个参数较多的函数给封装成参数较少的函数,因此对于上述find_if函数的问题,我们可以自定义一个含俩个参数的函数,然后通过bind函数进行封装,使之变成含一个参数的新函数(新函数会调用原来的函数),这样新函数就可以被find_if函数的第三个参数所使用了
2.bind的基本形式与使用
bind函数定义在头文件functional中,我们可以将bind函数看作一个通用的函数适配器,它的一般形式如下
auto newFun = bind(oldFun,arg_list);
参数oldFun是需要bind封装的源函数,newFun是封装了参数后的old_Fun,arg_list是一个逗号分割的参数列表,对应oldFun的参数,即当我们调用newFun是,它会调用oldFun并且会把参数列表中的参数传给oldFun
arg_list中会包含_n的名字,此类名字的参数又名”占位符”,因为其占据了newCallable的参数的位置,其中_n中的n表示它占据了new_Fun函数中的第几个参数。
函数的基本形式介绍完了,那么就进入整体,来为大家展示一个bind函数在解决find_if问题上是如何做的
实例如下
#include<iostream>
#include<vector>
#include<algorithm>
#include<functional>
using namespace std;
bool check_size(const int x,int sz)
{
return x > sz;
}
int main(void)
{
vector<int> v = {
1,2,3,4,5,6,7,8,9
};
int n = 5;
//有bind函数新封装生成的函数,其内部调用了check_size
auto new_check_size = bind(check_size,std::placeholders::_1,n);
auto it = find_if(v.begin(),v.end(),new_check_size);
cout<<"第一个大于n的数为:"<<*it<<endl;
return 0;
}
3.用bind重排源函数的参数顺序
用bind重排源函数的参数顺序只需将新函数与源函数的参数列表进行跌倒即可
实现代码如下
//假定源函数如下
bool oldFun(int a1,int a2);
//使用bind封装源函数如下
auto newFun = bind(old_Fun,_2,_1);
4.使用ref给源函数传递引用参数
如果我们想像lambda表达式一样传递引用,那么就得使用标准库中的ref函数,与其类似的是cref其生成的引用是const类型的
具体实例如下
#include<iostream>
#include<vector>
#include<algorithm>
#include<functional>
using namespace std;
bool check_size(const int x,int &sz)
{
//改变sz的值
sz = 6;
return x > sz;
}
int main(void)
{
vector<int> v = {
1,2,3,4,5,6,7,8,9
};
int n = 5;
//传递n的引用
auto new_check_size = bind(check_size,std::placeholders::_1,ref(n));
auto it = find_if(v.begin(),v.end(),new_check_size);
cout<<"n的值为为:"<<n<<endl;
return 0;
}