【Example】C++ 回调函数及 std::function 与 std::bind

简介: 回调函数是做为参数传递的一种函数,在早期C样式编程当中,回调函数必须依赖函数指针来实现。而后的C++语言当中,又引入了 std::function 与 std::bind 来配合进行回调函数实现。标准库中有大量函数应用到了回调函数,其中 std::sort 就是一个经典例子。

一,回调函数

 

回调函数的创建步骤大概为:

1,声明一个函数指针类型。

2,拟写使用回调函数的函数,将函数指针类型及变量名声明作为参数传递。

3,拟写符合函数指针类型的实现函数,将实现函数的指针作为参数传递给使用它的函数。

 

下面演示了一个最简单的回调函数定义及使用:

typedefint (*Calc)(inta, intb);
intCalcValue(inta, intb, constCalc&func) {
returnfunc(a, b);
}
intAdd(inta, intb) {
returna+b;
}
intmain()
{
inta=4;
intb=6;
intc=CalcValue(a, b, Add);
std::cout<<"Value: "<<c<<std::endl;
returnEXIT_SUCCESS;
}

可以看到,我们通过语法:

typedefint (*Calc)(inta, intb);

来定义了回调函数的指针类型,包括返回值类型、(*类型名)函数指针、参数表。

继而又定义并且实现了回调函数的使用者函数:

intCalcValue(inta, intb, constCalc&func) {
returnfunc(a, b);
}

再去定义并实现符合函数指针类型的实现函数:

intAdd(inta, intb) {
returna+b;
}

必须要注意的是,实现函数的类型必须要和函数指针的类型声明一致,也就是返回值和参数表(个数、类型)要完全一致。

这样就完成了一个简单且最基本的回调函数。

 

那么,回调函数是什么情况下使用的呢?

举个最经典的例子就是 std::sort,当你需要给一个存储有自定义结构体的 vector 进行排序时,编译器是无法知道如何对自定义结构体进行排序的。

这时候就需要实现一个回调函数来告诉编译器如何排序:

typedefstructDataPool {
intvalue=0;
intdate=0;
structDataPool(intv, intd) : value(v), date(d) {};
}DataPool;
boolSortCallBack(constDataPool&a, constDataPool&b) {
returna.value<b.value;
};
intmain()
{
vector<DataPool>vec;
vec.push_back(DataPool(2, 1));
vec.push_back(DataPool(5, 2));
vec.push_back(DataPool(1, 3));
std::sort(vec.begin(), vec.end(), SortCallBack);
returnEXIT_SUCCESS;
}

这样,就相当于自定义了 struct 的排序规则,自然编译器也可以使用 std::sort 对自定义 struct 进行排序操作。

 

二、std::function 与 std::bind

上面演示了最简单的回调函数创建及使用,然而,上面的代码却出现了一个局限性,就是:

如果需要去回调一个类成员函数,函数指针则无法指向类成员函数

在基本C样式面向过程编程当中,这种局限性并不那么明显甚至可以说不存在。但是到了C++当中,这种弊端就显而易见了,解决方式便是使用 std::function 与 std::bind 互相配合。

它们的头文件是:

#include <functional>

 

std::function

std::function 是一个模板类。作用是对C++中的可调用对象进行包装,例如普通函数、成员函数、模板函数、静态函数、lambda表达式等。

它的最基本的作用是,简化调用的复杂程度,统一调用的方式。如果代码中混杂着大量普通函数、模板函数、lambda,使用 std::function 是非常有必要的。

语法是:

【伪代码】std::function<returnType(argType, argType,...)>func;
【常规情况】std::function<int(int, int)>func;

可以看到,这个模板类当中对类型的声明方式是 < 返回值类型 ( 参数类型1, 参数类型2, ...) >。

 

你几乎可以拿它包装任何可调用对象,只需简单粗暴的将可调用对象作为右值赋值给它:

boolCompareInt(inta, intb) {
returna>b;
}
std::function<bool(int, int)>compareFunc=CompareInt;

那么如何使用它来调用类成员函数呢?这时就需要用到经常与 std::function 配合使用的 std::bind。

 

std::bind

它是一个基于模板的函数,顾明思意它的作用是绑定并返回一个 std::function 对象。

那么什么是“绑定”?它本身作为延迟计算的思想的一种实现,作为一个调用过程当中的转发者而存在,返回一个 std::function 对象

它与 std::function 不同的是,function 是模板类,bind 是模板函数,而 bind 返回的可调用对象可以直接给 function 进行包装并保存

 

为什么要进行“包装”与“转发”呢?

首先,不规范的解释是,function 的作用是包装,它可以包装类成员函数,但却无法生成类成员函数的可调用对象。而 std::bind 则是可以生成。

因此,function 与 bind 结合后,便成为了 C++ 中类成员函数作为回调函数的一种规范的实现方式。

 

std::bind 的语法是:

intAdd(inta, intb) {
returna+b;
}
/* --- 普通函数 --- */【伪代码】std::bind(&funcName, std::placeholders::_1, ...);
【常规情况】std::bind(&Add, std::placeholders::_1, std::placeholders::_2);
/* --- 类成员函数 --- */【伪代码】std::bind(&className::funcName, classPtr, std::placeholders::_1, ...);
【常规情况】std::bind(&BrainToolBox::Add, brain, std::placeholders::_1, std::placeholders::_2);

 

当用作普通函数的绑定时,第一个参数是可调用对象(普通函数、lambda等),而第二个参数开始对应可调用对象的参数表。

std::placeholders::_1 代表可调用对象的第一个参数,_2就代表第二个参数,依此类推。

 

当用作类成员函数的绑定时,第一个参数仍然是作为类成员的可调用对象引用,第二个参数则是对象的指针,而第三个参数开始对应可调用对象的参数表。

同样使用 std::placeholders::_* 依次向后推。

 

所以,与 std::function 相结合,便可以实现对类成员函数的调用:

classBrainToolBox{
public:
intAdd(inta, intb) { returna+b; };
};
intmain()
{
inta=4;
intb=6;
std::shared_ptr<BrainToolBox>brain=std::make_shared<BrainToolBox>();
std::function<int(int, int)>addFunc=std::bind(&BrainToolBox::Add, brain, std::placeholders::_1, std::placeholders::_2);
intc=addFunc(a, b);
std::cout<<"c Value: "<<c<<std::endl;
returnEXIT_SUCCESS;
}

 

对 std::bind 的额外注解:

1,调用指向非静态成员函数指针或指向非静态数据成员指针时,首参数必须是引用或指针(可以包含智能指针,如 std::shared_ptr 与 std::unique_ptr),指向将访问其成员的对象。

2,到 bind 的参数被复制或移动,而且决不按引用传递,除非包装于 std::ref 或 std::cref 。

3,允许同一 bind 表达式中的多重占位符(例如多个 _1 ),但结果仅若对应参数( u1 )是左值或不可移动右值才良好定义。

--- CPP Reference





====================================

芯片烤电池 C++ Example 2022-Spring Season Pass :

【Example】C++ 标准库常用容器全面概述

【Example】C++ 回调函数及 std::function 与 std::bind

【Example】C++ 运算符重载

【Example】C++ 标准库智能指针 unique_ptr 与 shared_ptr

【Example】C++ 接口(抽象类)概念讲解及例子演示

【Example】C++ 虚基类与虚继承 (菱形继承问题)

【Example】C++ Template (模板)概念讲解及编译避坑

【Example】C++ 标准库 std::thread 与 std::mutex

【Example】C++ 标准库多线程同步及数据共享 (std::future 与 std::promise)

【Example】C++ 标准库 std::condition_variable

【Example】C++ 用于编译时封装的 Pimpl 演示 (编译防火墙 Private-IMPL)

【Example】C++ 单例模式 演示代码 (被动模式、兼容VS2022编译)

====================================

相关文章
|
24天前
|
存储 安全 编译器
【C++ 17 新功能 std::visit 】深入解析 C++17 中的 std::visit:从原理到实践
【C++ 17 新功能 std::visit 】深入解析 C++17 中的 std::visit:从原理到实践
69 0
|
24天前
|
存储 JSON 安全
【C++ JSON库 json值的创建手段】深入探究C++中JSON对象定位与操作:从引用到回调函数
【C++ JSON库 json值的创建手段】深入探究C++中JSON对象定位与操作:从引用到回调函数
60 0
|
12天前
|
编译器 C语言 C++
【C++的奇迹之旅(二)】C++关键字&&命名空间使用的三种方式&&C++输入&输出&&命名空间std的使用惯例
【C++的奇迹之旅(二)】C++关键字&&命名空间使用的三种方式&&C++输入&输出&&命名空间std的使用惯例
|
16天前
|
人工智能 机器人 中间件
【C++】C++回调函数基本用法(详细讲解)
【C++】C++回调函数基本用法(详细讲解)
|
22天前
|
安全 程序员 C++
【C++ 基本知识】现代C++内存管理:探究std::make_系列函数的力量
【C++ 基本知识】现代C++内存管理:探究std::make_系列函数的力量
97 0
|
24天前
|
存储 安全 编译器
【C++ 函数设计的艺术】深挖 C++ 函数参数的选择 智能指针与 std::optional:最佳实践与陷阱
【C++ 函数设计的艺术】深挖 C++ 函数参数的选择 智能指针与 std::optional:最佳实践与陷阱
106 0
|
24天前
|
存储 安全 编译器
【C++ 17 泛型容器对比】C++ 深度解析:std::any 与 std::variant 的细微差别
【C++ 17 泛型容器对比】C++ 深度解析:std::any 与 std::variant 的细微差别
46 1
|
24天前
|
算法 编译器 C++
【C++ 泛型编程 中级篇】C++ 编译时技术:探索 if constexpr 和 std::enable_if
【C++ 泛型编程 中级篇】C++ 编译时技术:探索 if constexpr 和 std::enable_if
36 0
|
17天前
|
资源调度 Serverless 计算机视觉
高斯函数 Gaussian Function
**高斯函数,或称正态分布,以数学家高斯命名,具有钟形曲线特征。关键参数包括期望值μ(决定分布中心)和标准差σ(影响分布的宽度)。当μ=0且σ²=1时,分布为标准正态分布。高斯函数广泛应用于统计学、信号处理和图像处理,如高斯滤波器用于图像模糊。其概率密度函数为e^(-x²/2σ²),积分结果为误差函数。在编程中,高斯函数常用于创建二维权重矩阵进行图像的加权平均,实现模糊效果。
14 1
|
30天前
|
算法 Serverless C语言
CMake函数和宏(function和macro):使用函数和宏提高代码可读性
CMake函数和宏(function和macro):使用函数和宏提高代码可读性
30 1

热门文章

最新文章