通过手撕std::bind来了解其工作原理

std::bind 是 C++11 中的一个函数模板,用于创建一个可调用对象(函数对象或者函数指针)的绑定副本,其中一部分参数被固定为指定值,从而生成一个新的可调用对象。

std::bind 的原理可以简单概括如下:

  1. 创建绑定副本:

    当调用 std::bind 时,会生成一个新的可调用对象,其中一部分参数被绑定为指定值,剩余的参数保留为占位符(placeholder)。

  2. 占位符:

    std::bind 中,占位符使用 _1, _2, _3 等进行表示,分别代表可调用对象的第一个、第二个、第三个参数,以此类推。在创建绑定副本时,可以通过占位符指定参数的位置。

  3. 生成新的可调用对象:

    绑定副本可以作为一个新的可调用对象,可以直接调用,也可以传递给其他接受可调用对象的函数,从而实现部分参数固定的函数调用。

  4. 参数传递:

    在调用绑定副本时,传递给绑定副本的参数会替换掉占位符,从而形成最终的参数列表,并调用原始函数或函数对象。

  5. 绑定对象的生命周期:

    绑定副本会对传递给它的参数进行复制或者引用,因此绑定对象中的参数需要在绑定副本被调用时仍然有效。

以下是 std::bind 的一个简单示例:

cpp 复制代码
#include <iostream>
#include <functional>

void print_sum(int a, int b) {
    std::cout << a + b << std::endl;
}

int main() {
    auto f = std::bind(print_sum, 10, std::placeholders::_1);
    f(20); // 输出 30
    return 0;
}

在这个示例中,std::bind 生成了一个新的可调用对象 f,其中 print_sum 函数的第一个参数被绑定为 10,而第二个参数被保留为占位符 _1。当调用 f(20) 时,20 会替换占位符 _1,从而调用 print_sum(10, 20),输出 30。

下面,我们来实现一个简化版的std::bind,以更加深入的了解std::bind的工作原理:

cpp 复制代码
#include <iostream>
#include <functional>

// 定义一个模板类 MyBind,用于实现类似 std::bind 的功能
template <typename Func, typename... Args>
class MyBind {
public:
    // 构造函数,接受一个函数对象和一系列参数,这里需要用&&,以实现完美转发
    MyBind(Func&& func, Args&&... args)
        : func_(std::forward<Func>(func)), args_(std::forward<Args>(args)...) {}

    // 重载 () 运算符,用于调用绑定的函数对象
    template <typename... OtherArgs>
    auto operator()(OtherArgs&&... other_args) {
        // 调用 call_func 函数,将传递给它的参数与绑定的参数一起传递给原始函数对象
        return call_func(std::index_sequence_for<Args...>(), std::forward<OtherArgs>(other_args)...);
    }

private:
    // call_func 函数用于将传递给它的参数与绑定的参数一起传递给原始函数对象
    template <size_t... I, typename... OtherArgs>
    auto call_func(std::index_sequence<I...>, OtherArgs&&... other_args) {
        // 调用原始函数对象,传递绑定的参数和其他参数
        return func_(std::get<I>(args_)..., std::forward<OtherArgs>(other_args)...);
    }

    Func func_;                         // 原始的函数对象
    std::tuple<Args...> args_;          // 绑定的参数
};

// 辅助函数,用于创建 MyBind 对象
template <typename Func, typename... Args>
MyBind<Func, Args...> my_bind(Func&& func, Args&&... args) {
    return MyBind<Func, Args...>(std::forward<Func>(func), std::forward<Args>(args)...);
}

// 测试函数
void print_sum(int a, int b) {
    std::cout << a + b << std::endl;
}

int main() {
    auto f = my_bind(print_sum, 10, std::placeholders::_1); // 创建 MyBind 对象 f,绑定 print_sum 函数和参数
    f(20); // 调用绑定对象 f,传递参数 20,输出 30
    return 0;
}
相关推荐
J2虾虾4 分钟前
Java使用的可以使用的脚本执行引擎
java·开发语言·脚本执行
幻云20107 分钟前
Next.js指南:从入门到精通
开发语言·javascript·人工智能·python·架构
老马识途2.09 分钟前
java处理接口返回的json数据步骤 包括重试处理,异常抛出,日志打印,注意事项
java·开发语言
CCPC不拿奖不改名14 分钟前
网络与API:从HTTP协议视角理解网络分层原理+面试习题
开发语言·网络·python·网络协议·学习·http·面试
代码游侠24 分钟前
学习笔记——HC-SR04 超声波测距传感器
开发语言·笔记·嵌入式硬件·学习
superman超哥36 分钟前
Context与任务上下文传递:Rust异步编程的信息高速公路
开发语言·rust·编程语言·context与任务上下文传递·rust异步编程
橘颂TA36 分钟前
【剑斩OFFER】算法的暴力美学——leetCode 946 题:验证栈序列
c++·算法·leetcode·职场和发展·结构与算法
步达硬件37 分钟前
【Matlab】批量自定义图像处理
开发语言·matlab
闻缺陷则喜何志丹38 分钟前
【状态机动态规划】3686. 稳定子序列的数量|1969
c++·算法·动态规划·力扣·状态机动态规划
军军君0138 分钟前
Three.js基础功能学习七:加载器与管理器
开发语言·前端·javascript·学习·3d·threejs·三维