zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    如何使用RedisTemplate访问Redis数据结构
    redis 简单限流
    Caused by: org.apache.ibatis.binding.BindingException: Parameter '__frch_item_0' not found. Available parameters are [list]
    简单的根据权重随机数负载均衡算法
    后缀表达式(逆波兰表达式)计算器
    ASP.NET C# 如何在程序中控制IIS服务或应用程序池重启?
    iis7下url重写后,已存在的html不能访问了(未能执行URL)的解决方法
    IIS支持apk文件
    用vbs脚本远程登录批量交换机保存配置
    卷积神经网络入门案例-数字图像识别
  • 原文地址:https://www.cnblogs.com/xiaoshiwang/p/9678769.html
Copyright © 2011-2022 走看看