c/c++ 标准库 bind 函数详解

时间:2018-09-20
bind是一组用于函数绑定的模板。在对某个函数进行绑定时,可以指定部分参数或全部参数,也可以不指定任何参数,还可以调整各个参数间的顺序。这篇文章主要介绍了c/c++ 标准库 bind 函数 ,需要的朋友可以参考下

bind函数定义在头文件 functional 中。可以将 bind 函数看作一个通用的函数适配器,它接受一个可调用对象,生成一个新的可调用对象来“适应”原对象的参数列表。

bind函数:接收一个函数名作为参数,生成一个新的函数。

auto newCallable = bind(callbale, arg_list);

arg_list中的参数可能包含入_1, _2等,这些是新函数newCallable的参数。

在这篇博客lambda 表达式 介绍 中,讨论了find_if的第三个参数的问题,当时是用lambda表达式解决的,有了bind函数后,也可以用bind函数解决。

解决办法:bind(check_size, _1, sz)

auto idx = find_if(svec.begin(),svec.end(),bind(check_size, _1, 6));

其实,newCall= bind(check_size, _1, sz)返回了一个新的函数newCall ,这个newCall 只接受一个参数,正好满足find_if的要求。

•从find_if的角度来看,啊,newCall是含有一个参数的函数,OK,没问题。
•从程序猿的角度看,check_size是含有2个参数的函数,只是提前把sz(6)绑定到了newCall上了,
•当调用newCall(s),实际是调用了check_size(s, 6),相当于newCall也有2个参数,只是第二个参数有个默认值为6。newCall(const string &s, size_t sz = 6); ,所以调用newCall时,传递一个参数就够了。

注意:_1,_2等,是放在了命名空间placeholder中,所以要使用:

//_1,_n定在std::placeholders里面
using namespace std::placeholders;

bind参数用法:

//g是以个有2个参数的可调用对象
auto g = bind(func, a, b, _2, c, _1);//func是有5个参数的函数

调用g(X, Y), 等于 func(a, b, Y, c, X)

例子:

#include <iostream>
#include <vector>
#include <string>
#include <algorithm>
#include <functional>
using namespace std;
//_1,_n定在std::placeholders里面                        
using namespace std::placeholders;
bool check_size(const string &s, string::size_type sz){
 return s.size() >= sz;
}
bool shorter(const string &a, const string &b){
 return a.size() < b.size();
}
ostream& print(ostream& os, const string &s, const char &c){
 //c = ',';                                  
 return os << s << c;
}
int main(){
 /*                                      
 //用bind实现了和lambda一样的功能                       
 vector<string> svec{"aab","d","aa","bb","e","bbb"};              
 stable_sort(svec.begin(),svec.end(),[](const string &a, const string &b){   
   return a.size() < b.size();                        
  });                                     
 string::size_type sz = 3;                           
 auto idx = find_if(svec.begin(),svec.end(),bind(check_size, _1, sz));     
 cout << *idx << endl;                             
 idx = find_if(svec.begin(),svec.end(),[sz](const string &s){         
   return s.size() >= sz;                          
  });                                     
 cout << *idx << endl;                             
 */
 /*                                      
 //用bind改变原来函数的参数的位置                       
 //升序                                    
 vector<string> svec{"aab","d","aa","bb","e","bbb"};              
 sort(svec.begin(), svec.end(), shorter);                   
 for(auto const &s : svec){                          
  cout << s << " ";                              
 }                                       
 cout << endl;                                 
 //由于调换了shorter参数的位置,所以变成了降序                 
 sort(svec.begin(), svec.end(),bind(shorter, _2, _1));             
 for(auto const &s : svec){                          
  cout << s << " ";                              
 }                                       
 cout << endl;                                 
 */
 //bind引用,必须使用ref或者cref函数,把对象转化成引用,不能用&         
 ostream &os = cout;
 const char c = ' ';
 vector<string> svec{"aab","d","aa","bb","e","bbb"};
 for_each(svec.begin(),svec.end(),[&os, c](const string &s){
   os << s << c;
  });
 os << endl;
 for_each(svec.begin(),svec.end(),bind(print, ref(os), _1, cref(c)));
 os << endl;
 cout << c << endl;
}

总结

以上所述是小编给大家介绍的c/c++ 标准库 bind 函数详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!