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

Posted xiaoshiwang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了c/c++ 标准库 bind 函数 详解相关的知识,希望对你有一定的参考价值。

标准库 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++ 学习互助QQ群:877684253

本人微信:xiaoshitou5854

以上是关于c/c++ 标准库 bind 函数 详解的主要内容,如果未能解决你的问题,请参考以下文章

C语言正则表达式详解 regcomp() regexec() regfree()详解

C/C++网络编程详解(Windows版)

C/C++: C++可调用对象详解

linux 进程间通信 dbus-glib实例详解四(上) C库 dbus-glib 使用(附代码)(编写接口描述文件.xml,dbus-binding-tool工具生成绑定文件)(列集散集函数)

c/c++标准库中的文件操作总结

C/C++C++11新特性:std::bind