返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++11 中的std::function和std::bind详解
  • 702
分享到

C++11 中的std::function和std::bind详解

2024-04-02 19:04:59 702人浏览 八月长安
摘要

目录1. 可调用对象2. std::function3. std::bind3.1 std::bind绑定普通函数3.2 std::bind绑定一个成员函数3.3 绑定一个引用参数4

1. 可调用对象

可调用对象有一下几种定义:

  • 是一个函数指针,参考 c++ 函数指针和函数类型;
  • 是一个具有operator()成员函数的类的对象;
  • 可被转换成函数指针的类对象;
  • 一个类成员函数指针;

C++中可调用对象的虽然都有一个比较统一的操作形式,但是定义方法五花八门,这样就导致使用统一的方式保存可调用对象或者传递可调用对象时,会十分繁琐。C++11中提供了std::function和std::bind统一了可调用对象的各种操作。

不同类型可能具有相同的调用形式,如:


// 普通函数
int add(int a, int b){return a+b;} 
// lambda表达式
auto mod = [](int a, int b){ return a % b;}
// 函数对象类
struct divide{
    int operator()(int denominator, int divisor){
        return denominator/divisor;
    }
};

上述三种可调用对象虽然类型不同,但是共享了一种调用形式:


int(int ,int)

std::function就可以将上述类型保存起来,如下:


std::function<int(int ,int)>  a = add; 
std::function<int(int ,int)>  b = mod ; 
std::function<int(int ,int)>  c = divide(); 

2. std::function

  • std::function 是一个可调用对象包装器,是一个类模板,可以容纳除了类成员函数指针之外的所有可调用对象,它可以用统一的方式处理函数、函数对象、函数指针,并允许保存和延迟它们的执行。
  • 定义格式:std::function<函数类型>。
  • std::function可以取代函数指针的作用,因为它可以延迟函数的执行,特别适合作为回调函数使用。它比普通函数指针更加的灵活和便利。

3. std::bind

可将std::bind函数看作一个通用的函数适配器,它接受一个可调用对象,生成一个新的可调用对象来“适应”原对象的参数列表。

std::bind将可调用对象与其参数一起进行绑定,绑定后的结果可以使用std::function保存。std::bind主要有以下两个作用:

  • 将可调用对象和其参数绑定成一个防函数;
  • 只绑定部分参数,减少可调用对象传入的参数。

3.1 std::bind绑定普通函数

    
    double my_divide (double x, double y) {return x/y;}
    auto fn_half = std::bind (my_divide,_1,2);  
    std::cout << fn_half(10) << '\n';                        // 5
  • bind的第一个参数是函数名,普通函数做实参时,会隐式转换成函数指针。因此std::bind (my_divide,_1,2)等价于std::bind (&my_divide,_1,2);
  • _1表示占位符,位于<functional>中,std::placeholders::_1;

3.2 std::bind绑定一个成员函数

    
    struct Foo {
        void print_sum(int n1, int n2)
        {
            std::cout << n1+n2 << '\n';
        }
        int data = 10;
    };
    int main() 
    {
        Foo foo;
        auto f = std::bind(&Foo::print_sum, &foo, 95, std::placeholders::_1);
        f(5); // 100
    }
  • bind绑定类成员函数时,第一个参数表示对象的成员函数的指针,第二个参数表示对象的地址。
  • 必须显示的指定&Foo::print_sum,因为编译器不会将对象的成员函数隐式转换成函数指针,所以必须在Foo::print_sum前添加&;
  • 使用对象成员函数的指针时,必须要知道该指针属于哪个对象,因此第二个参数为对象的地址 &foo;

3.3 绑定一个引用参数

默认情况下,bind的那些不是占位符的参数被拷贝到bind返回的可调用对象中。但是,与lambda类似,有时对有些绑定的参数希望以引用的方式传递,或是要绑定参数的类型无法拷贝。


#include <iOStream>
#include <functional>
#include <vector>
#include <alGorithm>
#include <sstream>
using namespace std::placeholders;
using namespace std;
ostream & print(ostream &os, const string& s, char c)
{
    os << s << c;
    return os;
}
int main()
{
    vector<string> Words{"helo", "world", "this", "is", "C++11"};
    ostringstream os;
    char c = ' ';
    for_each(words.begin(), words.end(), 
                   [&os, c](const string & s){os << s << c;} );
    cout << os.str() << endl;
    ostringstream os1;
    // ostream不能拷贝,若希望传递给bind一个对象,
    // 而不拷贝它,就必须使用标准库提供的ref函数
    for_each(words.begin(), words.end(),
                   bind(print, ref(os1), _1, c));
    cout << os1.str() << endl;
}

4. 指向成员函数的指针

通过下面的例子,熟悉一下指向成员函数的指针的定义方法。

    
    #include <iostream>
    struct Foo {
        int value;
        void f() { std::cout << "f(" << this->value << ")\n"; }
        void g() { std::cout << "g(" << this->value << ")\n"; }
    };
    void apply(Foo* foo1, Foo* foo2, void (Foo::*fun)()) {
        (foo1->*fun)();  // call fun on the object foo1
        (foo2->*fun)();  // call fun on the object foo2
    }
    int main() {
        Foo foo1{1};
        Foo foo2{2};
        apply(&foo1, &foo2, &Foo::f);
        apply(&foo1, &foo2, &Foo::g);
    }
  • 成员函数指针的定义:void (Foo::*fun)(),调用是传递的实参: &Foo::f;
  • fun为类成员函数指针,所以调用是要通过解引用的方式获取成员函数*fun,即(foo1->*fun)();

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!

--结束END--

本文标题: C++11 中的std::function和std::bind详解

本文链接: https://lsjlt.com/news/137729.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C++11 中的std::function和std::bind详解
    目录1. 可调用对象2. std::function3. std::bind3.1 std::bind绑定普通函数3.2 std::bind绑定一个成员函数3.3 绑定一个引用参数4...
    99+
    2024-04-02
  • C++11 std::function和std::bind如何使用
    这篇文章主要介绍了C++11 std::function和std::bind如何使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++11 std::function和std::bind如...
    99+
    2023-07-05
  • C++11中std::function与std::bind的用法实例
    目录关于std::function 的用法:关于std::bind 的用法:附:std::function与std::bind双剑合璧总结关于std::function 的用法: 其...
    99+
    2024-04-02
  • C++11std::function和std::bind的使用示例详解
    目录概述可调用对象std::functionstd::function函数原型std::function的主要作用std::function的优缺点std::bindstd::bin...
    99+
    2023-03-19
    C++11 std::function和std::bind C++11 std::function和std::bind使用
  • C++ std::function详解
    目录一、介绍二、实例三、与std::bind的区别总结一、介绍 std::function是函数模板类(是一个类)。包含在#include <functional> 中。...
    99+
    2024-04-02
  • C++11中std::function基础用法详解
    目录一、std::function基本介绍二、进阶使用方法2.1 与智能指针相结合2.2 存储成员函数指针2.3 存储std::bind三、注意tipsstd::function是C...
    99+
    2023-05-18
    C++11 std::function用法 C++11 std::function C++ std::function
  • C++ std::bind用法详解
    一、介绍 C++11中提供了std::bind。bind()函数的意义就像它的函数名一样,是用来绑定函数调用的某些参数的。 bind的思想实际上是一种延迟计算的思想,将可调用对象保存...
    99+
    2024-04-02
  • C++ std::function的用法详解
    类模版std::function是一种通用、多态的函数封装。std::function的实例可以对任何可以调用的目标实体进行存储、复制、和调用操作,这些目标实体包括普通函数、Lamb...
    99+
    2024-04-02
  • C++11 学习笔记之std::function和bind绑定器
    std::function       C++中的可调用对象虽然具有比较统一操作形式(除了类成员指针之外,都是后面加括号进行调用),...
    99+
    2024-04-02
  • 浅谈C++11的std::function源码解析
    目录1、源码准备2、std::function简介3、源码解析3.1、std::function解析3.2、std::_Function_handler解析3.3、_Any_data...
    99+
    2024-04-02
  • 解析C++11的std::ref、std::cref源码
    目录1、源码准备2、std::ref和std::cref的作用3、std::ref相关源码解析3.1、std::ref解析3.2、std::reference_wrapper解析3....
    99+
    2024-04-02
  • C++11中std::ref和std::cref的作用是什么
    这篇文章给大家分享的是有关C++11中std::ref和std::cref的作用是什么的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。1、源码准备本文是基于gcc-4.9.0的源代码进行分析,std::ref和st...
    99+
    2023-06-15
  • 详解C++11的std::addressof源码解析
    目录1、源码准备2、std::addressof简介3、std::addressof源码解析4、总结1、源码准备 本文是基于gcc-4.9.0的源代码进行分析,std::addre...
    99+
    2024-04-02
  • C++20中的std::span详解
    span就是一个连续对象存储的观察者。类似std::string_view是string的观察者。连续的存储,不一定是数组。例如: zero(char (&arr)...
    99+
    2023-03-06
    C++20 std::span C++ std
  • c++11中std::move函数的使用
    C++11在运行期有所增强,通过增加核心的右值引用机制来改善临时对象导致的效率低下的问题。C++临时对象引入了多余的构造、析构及其内部资源的申请释放函数调用,导致程序运行时性能受损,...
    99+
    2024-04-02
  • 浅谈C++11的std::mem_fn源码解析
    目录1、源码准备2、通过一个简单的例子来了解std::mem_fn的作用3、std::mem_fn源码解析3.1、std::mem_fn解析3.2、std::_Mem_fn解析3.3...
    99+
    2024-04-02
  • C++ 函数模板与 std::function 的比较和应用?
    函数模板和 std::func++tion 都是 c++ 中表示函数的方法,它们各有优劣:函数模板:静态类型安全、性能优异,但灵活性较低,不能存储动态函数对象。std::function...
    99+
    2024-04-24
    关键词: c++ 函数模板 c++
  • C++中std::allocator的使用案例详解
    标准库中包含一个名为allocator的类,允许我们将分配和初始化分离。使用allocator通常会提供更好的性能和更灵活的内存管理能力。    ...
    99+
    2024-04-02
  • C++11的std::future对象如何使用
    这篇“C++11的std::future对象如何使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C++11的std::fu...
    99+
    2023-07-05
  • 如何进行c++11中std::move函数的使用
    这篇文章给大家介绍如何进行c++11中std::move函数的使用,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。C++11在运行期有所增强,通过增加核心的右值引用机制来改善临时对象导致的效率低下的问题。C++临时对象引...
    99+
    2023-06-29
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作