当前位置: 首页 > news >正文

【C++11 —— 包装器】

C++11 —— 包装器

  • 包装器
    • function包装器
      • function包装器介绍
      • function包装器统一类型
      • function包装器的意义
    • bind包装器
      • bind包装器介绍
      • bind包装器绑定固定参数
      • bind包装器调整传参顺序
      • bind包装器的意义

包装器

function包装器

function包装器介绍

function包装器 也叫作适配器。C++ 中的 function 本质是一个类模板,也是一个包装器。这些包装器主要用于提供一致且适合的接口以便于处理各种可调用对象,如函数、函数对象、lambda表达式等。

std::function的原型如下:

template< class R, class... Args >
class function<R(Args...)>;

其中:

  • R: 是被调用函数的返回类型。
  • Args: 是被调用函数的参数类型列表。
std::function<int(int, int)> func; // func 可以存储接受两个 int 参数并返回 int 的可调用对象

function包装器统一类型

std::function是一个类模板,它提供了一种统一的方式来存储和调用各种可调用对象,包括:

  • 普通函数
  • Lambda 表达式
  • 函数指针
  • 函数对象(重载了 operator() 的类)
  • 成员函数指针
  • 数据成员指针

通过指定模板参数,std::function可以处理具有特定返回类型和参数类型列表的可调用对象。它使用模板转换构造函数来接收被包装的函数对象。

下面这段代码,useF是一个函数模板,接受两个模板参数FT
F表示可调用对象的类型,T表示参数的类型
使用可调用对象f,传入参数x,同时声明静态整形变量count以记录函数被调用的次数。

template<class F, class T>
T useF(F f, T x)
{static int count = 0;cout << "count:" << ++count << endl;cout << "count:" << &count << endl;return f(x);
}
double f(double i)
{return i / 2;
}
struct Functor
{double operator()(double d){return d / 3;}
};
int main()
{// 函数名cout << useF(f, 11.11) << endl;// 函数对象cout << useF(Functor(), 11.11) << endl;// lamber表达式cout << useF([](double d)->double { return d / 4; }, 11.11) << endl;return 0;
}

在这里插入图片描述
可以看到这里的count都是1,并且这里的地址也都不相同,说明这里的useF函数模板被实例化了三份。

但其实这里没必要实例化三份出来,虽然传入的可调用对象的类型不相同,但是外层显示看到的参数和返回值都是相同的,此时就可以使用包装器来进行上层封装,来优化效率,只实例化一个模板。

int main()
{// 函数名cout << useF(f, 11.11) << endl;// 函数对象cout << useF(Functor(), 11.11) << endl;// lamber表达式cout << useF([](double d)->double { return d / 4; }, 11.11) << endl;cout << endl << endl;function<double(double)> func1 = f;cout << useF(func1, 11.11) << endl;function<double(double)> func2 = Functor();cout << useF(func2, 11.11) << endl;function<double(double)> func3 = [](double d)->double {return d / 4; };cout << useF(func3, 11.11) << endl;return 0;
}

进行包装后可见,打印出来的count的每次地址都是相同的,并且count被累加到3,说明function做到了类型的统一,并且让其调用了3次。
在这里插入图片描述

使用 std::function的一些注意事项:

  1. 转换后的 std::function 对象的参数类型必须能转换为可调用对象的参数类型。
  2. 可调用对象的返回类型必须能转换为 std::function 对象的返回类型。
  3. 如果尝试调用一个存储了空可调用对象的 std::function,会抛出 std::bad_function_call 异常。

function包装器的意义

C++中存在多种可调用对象,包括普通函数、仿函数和lambda表达式。使用包装器(如std::function)可以将这些不同的可调用对象统一为一种类型,从而简化调用和管理。
比如下面的代码,在调用的时候就会显得比较凌乱,不统一,不符合C++11的编程习惯:


//包装器
#include <functional>int f(int a, int b)
{return a + b;
}struct F
{int operator()(int a,int b){return a + b;}
};int main()
{cout << f(1, 2) << endl;cout << "------------------" << endl;F f1;cout << f1(2, 3) << endl;cout << "------------------" << endl;cout << F()(3, 4) << endl;cout << "------------------" << endl;cout << [](int a, int b) {return a + b; }(4, 5) << endl;cout << "------------------" << endl;return 0;
}

在这里插入图片描述

所以使用包装器,可以做到接口的统一化,同时function也同时提供了类型安全的接口。通过定义返回类型和参数类型,编译器能够在编译的时候检查类型一致性,避免运行时错误。
包装器也减少了模板参数化的数量,从而提高了编译效率,通过将相同特征标的可用调用对象统一为一个类型,避免了多次实例化的问题!

	// 使用 std::function 包装普通函数 ffunction<int(int, int)> func1 = f; // func1 是一个可调用对象,类型为 int(int, int)cout << func1(1, 2) << endl; // 调用 func1,输出 3(1 + 2)cout << "------------------" << endl;// 使用 std::function 包装临时的 F 对象function<int(int, int)> func2 = F(); // 创建 F 的临时对象并包装cout << func2(2, 3) << endl; // 调用 func2,输出 5(2 + 3)cout << "------------------" << endl;// 创建 F 的实例 f1,并使用 std::function 包装它F f1; // 实例化 Ffunction<int(int, int)> func3 = f1; // 将 f1 包装到 func3 中cout << func3(3, 4) << endl; // 调用 func3,输出 7(3 + 4)cout << "------------------" << endl;// 使用 std::function 包装 lambda 表达式function<int(int, int)> func4 = [](int a, int b) { return a + b; }; // lambda 表达式cout << func4(4, 5) << endl; // 调用 func4,输出 9(4 + 5)cout << "------------------" << endl;

在这里插入图片描述

bind包装器

bind包装器介绍

bind也是一种函数包装器,也叫做适配器。它可以接受一个可调用对象,生成一个新的可调用对象来“适应”原对象的参数列表,C++中的bind本质是一个函数模板。

bind函数模板的原型如下:

template<typename F, typename... Args>
auto bind(F&& f, Args&&... args);

模板参数说明:

  • F 是可调用对象的类型。
  • Args 是绑定的参数类型,可以是值或占位符。

bind包装器绑定固定参数

可以使用 std::bind 将某些参数固定,从而生成一个新的可调用对象。例如:

#include <iostream>
#include <functional>void printSum(int a, int b, int c) {std::cout << "Sum: " << (a + b + c) << std::endl;
}int main() {// 绑定固定参数auto boundFunc = std::bind(printSum, 1, 2, std::placeholders::_1);// 调用时只需要提供最后一个参数boundFunc(3); // 输出: Sum: 6return 0;
}

在这里插入图片描述
在这个示例中,printSum 函数的前两个参数被固定为 1 和 2,而第三个参数使用占位符 _1,在调用boundFunc时提供。

bind包装器调整传参顺序

std::bind 还可以通过占位符调整参数的顺序。例如:

#include <iostream>
#include <functional>void printValues(int a, int b, int c) {std::cout << "Values: " << a << ", " << b << ", " << c << std::endl;
}int main() {// 调整参数顺序auto boundFunc = std::bind(printValues, std::placeholders::_2, 42, std::placeholders::_1);// 调用时提供两个参数boundFunc(10, 20); // 输出: Values: 20, 42, 10return 0;
}

在这里插入图片描述
在上面的代码中,printValues 函数的参数顺序被调整,第二个参数固定为 42,而 _1 和 _2 分别对应调用时提供的第一个和第二个参数。

bind包装器的意义

std::bind 是 C++11 中的一个函数适配器,它允许将可调用对象与部分参数绑定,从而生成新的可调用对象,简化函数调用并提高代码可读性。通过绑定固定参数和调整参数顺序,std::bind 提供了灵活性,使得在回调函数和事件处理等场景中,开发者能够更方便地管理和使用各种可调用对象。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 信息安全工程师(1)计算机网络分类
  • ubuntu下手工编译安装 6.* 最新内核
  • leetcode 146.LRU缓存
  • Encountered error while trying to install package.> lxml
  • VS Code 配置 Rust-Analyzer 报错
  • web渗透—RCE
  • SQL Server 语句日期格式查找方法
  • HT338 2x50W D类立体声音频功放
  • Android 测试机
  • 基于微信小程序的图书馆预约占座系统
  • 计算机毕业设计 自习室座位预约系统的设计与实现 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试
  • 【鸿蒙HarmonyOS NEXT】页面之间相互传递参数
  • 复旦:EoT下Muti-agentllm曾带给我的启发
  • 【pytorch】【onnx部署】系列学习文章目录
  • apache文件共享和访问控制
  • 时间复杂度分析经典问题——最大子序列和
  • [译]CSS 居中(Center)方法大合集
  • Android 初级面试者拾遗(前台界面篇)之 Activity 和 Fragment
  • axios请求、和返回数据拦截,统一请求报错提示_012
  • C++回声服务器_9-epoll边缘触发模式版本服务器
  • conda常用的命令
  • java2019面试题北京
  • NSTimer学习笔记
  • Redux 中间件分析
  • Sass Day-01
  • Spring技术内幕笔记(2):Spring MVC 与 Web
  • SQLServer插入数据
  • Storybook 5.0正式发布:有史以来变化最大的版本\n
  • weex踩坑之旅第一弹 ~ 搭建具有入口文件的weex脚手架
  • 对话:中国为什么有前途/ 写给中国的经济学
  • 给github项目添加CI badge
  • 基于 Ueditor 的现代化编辑器 Neditor 1.5.4 发布
  • 使用SAX解析XML
  • 我看到的前端
  • 想使用 MongoDB ,你应该了解这8个方面!
  • 云栖大讲堂Java基础入门(三)- 阿里巴巴Java开发手册介绍
  • 7行Python代码的人脸识别
  • raise 与 raise ... from 的区别
  • ​VRRP 虚拟路由冗余协议(华为)
  • ###51单片机学习(2)-----如何通过C语言运用延时函数设计LED流水灯
  • #NOIP 2014# day.2 T2 寻找道路
  • ()、[]、{}、(())、[[]]命令替换
  • (2009.11版)《网络管理员考试 考前冲刺预测卷及考点解析》复习重点
  • (2022 CVPR) Unbiased Teacher v2
  • (4)Elastix图像配准:3D图像
  • (C语言)fgets与fputs函数详解
  • (Forward) Music Player: From UI Proposal to Code
  • (js)循环条件满足时终止循环
  • (八)Docker网络跨主机通讯vxlan和vlan
  • (更新)A股上市公司华证ESG评级得分稳健性校验ESG得分年均值中位数(2009-2023年.12)
  • (回溯) LeetCode 131. 分割回文串
  • (蓝桥杯每日一题)平方末尾及补充(常用的字符串函数功能)
  • (转)Google的Objective-C编码规范
  • (转)jdk与jre的区别
  • .bat批处理(十一):替换字符串中包含百分号%的子串