C++std::function和std::bind()的概念

news/2025/2/19 17:53:14/

std::function: 一个通用的函数封装器,它允许你存储和调用任何可以被调用的东西,例如函数、函数指针、函数对象、Lambda 表达式等。

std::bind: 用于创建函数对象。一个可调用对象的绑定版本,可以提前绑定某些参数,稍后调用时只需提供剩余的参数。
在某些情况下,你可以将它们结合使用,例如,如果你想创建一个可调用对象,该对象包含绑定的参数,然后将其存储在 std::function 中。这在某些情况下可以提高代码的可读性和灵活性。

示例:

#include <iostream>
#include <functional>int add(int a, int b) {return a + b;
}int main() {// 创建一个绑定了参数的可调用对象,绑定参数为2,// 占位符_1表示稍后调用时传入的第1个参数放在_1这个位置std::function<int(int)> func = std::bind(add, 2, std::placeholders::_1);// 调用 std::functionint result = func(3); // 这里相当于调用 add(2, 3)std::cout << "Result: " << result << std::endl;return 0;
}

将std::function与std::bind一起使用可以提供一些好处,尤其在以下情况下:

  • 延迟参数绑定:你可以使用std::bind在创建std::function对象时部分绑定参数,然后稍后再提供其余的参数。这使得你可以在调用可调用对象时动态地确定一些参数值。

  • 增强可读性:std::bind允许你清晰地指定参数的绑定方式,这可以提高代码的可读性,特别是当你处理复杂的函数签名时。

  • 灵活性:结合使用std::function和std::bind可以让你更容易地操作和传递可调用对象。这对于实现通用接口或处理回调函数时特别有用。

  • 减少代码冗余:如果你需要多次调用具有相同部分绑定的可调用对象,结合使用std::function和std::bind可以减少重复的绑定代码。

在不需要提前绑定参数的情况下,可以不使用std::bind()。在C++11引入Lambda表达式后,使用更加灵活和方便。以下是一个示例,演示如何使用 std::function 存储一个普通函数或Lambda表达式

Copy code
#include <iostream>
#include <functional>// 普通函数
int add(int a, int b) {return a + b;
}int main() {// 使用 std::function 存储普通函数std::function<int(int, int)> func1 = add;// 使用 Lambda 表达式std::function<int(int, int)> func2 = [](int a, int b) {return a + b;};int result1 = func1(2, 3); // 调用 add 函数int result2 = func2(4, 5); // 调用 Lambda 表达式std::cout << "Result 1: " << result1 << std::endl;std::cout << "Result 2: " << result2 << std::endl;return 0;
}

考虑这种情况:可调用对象是类的普通成员函数

  • 成员函数是跟具体的对象绑定的,它可以操作该对象中的其他成员变量或者调用其他成员函数。成员函数可以做到分辨哪个对象在调用我是因为有隐藏this指针指向当前调用该函数的对象。
  • 比如obj.funcAdd(1,2),效果实际上是Obj::funcAdd(&obj, 1, 2);
  • 因此如果可调用对象是普通成员函数,我们要绑定一个参数,即指向调用对象的指针。

因此,比较推荐std::function与std::bind()配合使用,或者用lambda表达式

  • 当在一个成员函数中创建函数指针指向另一个普通成员函数时,绑定参数为this

    #include <iostream>
    #include <functional>class MyClass {
    public:void func1(int value) {std::cout << "func1: " << value << std::endl;}void func2(int value) {std::cout << "func2: " << value << std::endl;}void doSome() {// 使用 std::bind 创建 std::function 对象,将其绑定到 func2std::function<void(MyClass*, int)> functionPtr = std::bind(&MyClass::func2, this, std::placeholders::_2);// 调用函数指针,传入对象实例指针和参数functionPtr(42);}
    };int main() {MyClass obj;obj.doSome();return 0;
    }
    

    当然也可以手动传入this
    比如:std::bind(&MyClass::func2, std::placeholders::_1, std::placeholders::_2);
    调用:functionPtr(this, 42);

  • 利用lambda表达式作为可调用对象

    #include <iostream>
    #include <functional>class MyClass {
    public:void func1(int value) {std::cout << "func1: " << value << std::endl;}void func2(int value) {std::cout << "func2: " << value << std::endl;}void doSome() {// 使用 Lambda 表达式捕获 this 指针std::function<void(int)> functionPtr = [this](int value) {MemberFunction2(value);};// 调用函数指针,不需要传递 this 指针functionPtr(42);}
    };int main() {MyClass obj;obj.doSome();return 0;
    }
    
  • 还有一种简单的情况:如果是在想存储某个对象的某个成员函数的指针,则需要传入该对象的地址

    #include <iostream>
    #include <functional>class MyClass {
    public:void func(int value) {std::cout << "func: " << value << std::endl;}
    };int main() {MyClass obj;// 创建 std::function 对象,提前绑定&obj,如果不绑定则需要手动传入std::function<void(int)> funcPtr = std::bind(&MyClass::func, &obj, std::placeholders::_1);funcPtr(42);return 0;
    }
    

http://www.ppmy.cn/news/1090471.html

相关文章

Alibaba Canal 使用记录

项目中使用 canal 来同步数据到 Elasticsearch, 遇到很多问题&#xff0c;做一下记录&#xff1a; 版本问题&#xff1a; 1. 解析binlog出错 &#xff0c;表现为 limit excceed&#xff1a;xx 目前使用 mariadb 10.9.7/10.10.6 canal 1.1.6 hotfix &#xff0c;在这个版本组…

华为OD机试 - 关联子串 - 滑动窗口(Java 2023 B卷 100分)

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路五、Java算法源码六、效果展示1、输入2、输出3、说明4、换个思路 华为OD机试 2023B卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A…

SpringBoot 集成 Canal 实现监听MySQL表数据

SpringBoot 集成 Canal 准备工作什么是 CanalCanal 在 Spring Boot 中的作用和优势准备工作安装和配置 MySQL 数据库 安装Canal项目集成导入依赖添加配置信息创建监听类测试 准备工作 什么是 Canal Canal 是阿里巴巴开源的基于数据库增量日志解析的数据同步和订阅组件&#x…

Milvus以及Web UI 安装

向量数据库懂的都懂 版本数据 [rootiZ7xv7q4im4c48qen2do2bZ project]# cat /etc/redhat-release CentOS Stream release 9 [rootiZ7xv7q4im4c48qen2do2bZ project]# docker version Client: Docker Engine - CommunityVersion: 24.0.5API version: 1.43Go v…

10 种常用的字符串方法

10 种常用的字符串方法 1.concat() 字符串拼接 const str1 12345678;const str2 abcdefgh;const str3 -【】&#xff1b;‘;console.log(str1.concat(str2,str3))//12345678abcdefgh-【】&#xff1b;‘ 2.includes() 判断字符串中是否包含指定值&#xff0c;返回布尔值…

解决Echarts中双坐标轴分割错位问题

1、处理函数 /*** Description 刻度最大值* date 2023-08-30* param {any} isNaN(maxValue/1* returns {any}*/ export const getYAxisMax (maxValue): number > {if (isNaN(maxValue / 1) || maxValue / 1 < 10) {return 10;}const max: any Math.ceil(maxValue) ;c…

JavaScript中包含对象的数组去重

一.数组遍历 function Uniarray(array) {// 接收去重后的数组let resArr [];// 遍历数组for (let i 0; i < array.length; i) {let isFind false;// 检查当前元素是否已存在于结果数组中for (let j 0; j < resArr.length; j) {// 使用严格相等运算符&#xff08;&am…

protobuf安装及简单使用

protobuf简单介绍和ubuntu 16.04环境下安装教程&#xff1a;https://pythonjishu.com/rgdzjkxgoyicrhu/ Protocol Buffers使用指南&#xff1a;https://blog.csdn.net/jarvanxy/article/details/132256759