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

C++11 包装器function

文章首发公众号:iDoitnow

C++提供了多个包装器,它们主要是为了给其他编程接口提供更一致或更合适的接口。C++11提供了多个包装器,这里我们重点了解一下包装器function

对于function, C++ 参考手册给出的定义为:

类模板 std::function 是通用多态函数封装器。 std::function 的实例能存储、复制及调用任何可调用 (Callable) 目标——函数、 lambda 表达式、 bind 表达式或其他函数对象,还有指向成员函数指针和指向数据成员指针。

C++11为什么要引入function

我们先看一个例子:

#include <iostream>template <typename T, typename F>
T use_f(T v, F f) {static int count = 0;count++;std::cout << " use_f count = " << count << ", &count = " << &count<< std::endl;return f(v);
}class Fp {private:double z_;public:Fp(double z = 1.0) : z_(z) {}double operator()(double p) { return z_ * p; }
};class Fq {private:double z_;public:Fq(double z = 1.0) : z_(z) {}double operator()(double q) { return z_ + q; }
};double dub(double x) { return 2.0 * x; }
double square(double x) { return x * x; }
int main() {using std::cout;using std::endl;double y = 1.21;cout << "Function pointer dub:\n";/*dub是函数名(函数名是一个指针),因此参数F对应的类型为double(*)double,即一个指向(接受一个double参数并返回一个double的)函数指针*/cout << " " << use_f(y, dub) << endl; cout << "Function pointer square:\n";cout << " " << use_f(y, square) << endl;//同上cout << "Function object Fp:\n";//Fp(5.0)是一个参数对象,F对应的类型为Fpcout << " " << use_f(y, Fp(5.0)) << endl;cout << "Function object Fq:\n";//Fq(5.0)是一个参数对象,F对应的类型为Fqcout << " " << use_f(y, Fq(5.0)) << endl;cout << "Lambda expression 1:\n";//此处是一个lambda表达式,F对应的类型为该lambda表达式使用的类型cout << " " << use_f(y, [](double u) { return u * u; }) << endl;cout << "Lambda expression 2:\n";此处是一个lambda表达式,F对应的类型为该lambda表达式使用的类型cout << " " << use_f(y, [](double u) { return u + u / 2.0; }) << endl;return 0;
}

程序通过一个模板函数use_f使用参数f表示调用的类型,然后将f(v)返回。在主函数中我们6次调用模板函数,对于前两个调用的use_f为同一个实例化。后面四个,每一个都有其对应use_f的实例化。实际上,上述代码的运行结果如下:

Function pointer dub:use_f count = 1, &count = 0x555555756140
2.42
Function pointer square:use_f count = 2, &count = 0x555555756140
1.4641
Function object Fp:use_f count = 1, &count = 0x555555756144
6.05
Function object Fq:use_f count = 1, &count = 0x555555756148
6.21
Lambda expression 1:use_f count = 1, &count = 0x555555756138

从运行结果中,我们可以看出,在这六次调用中use_f的实例化了5次。使用模板函数,看似统一了操作形式,但其对于不同类型的F对模板函数都要进行一次实例化,这大大增加了编译的时长,并使头文件也增大,同时也降低了代码的执行效率。为了解决这类问题,我们首先能想到的解决办法就是:降低use_f的实例化的次数,理想的情况下是:在这6次循环调用的时候,调用同一个use_f的实例。针对上述例子,根据代码注释的分析,如果我们能将这6次调用中模板函数中F的类型保持统一,就可以像第一、二次调用的情况类似,使这六次调用同一个use_f的实例成为可能。

针对例子中的函数指针、函数对象和lambda表达式,它们有一个共同的特征:都是接受一个double参数并返回一个double值。也就是它们的调用特征标(它们的特征标都是double(double))相同。这便是function解决这个问题的关键。【:调用特征标是由返回类型和参数类型列表决定的,其格式为:返回类型(参数类型列表),其中每个参数类型用逗号分隔。】

因此,C++11引入了function包装器。function包装器可以简单理解为一个接口,它可以将特征标相同的函数指针、函数对象和lambda表达式等统一定义为一类特殊的对象。

function的用法

包装器function的本质是一个模板,它是在头文件functional中声明,其使用方法如下:

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

其中R为被调用函数的返回类型,Args为被调用函数的形参。存储的可调用对象被称为std::function的目标,如果目标为空,则调用空的function会导致抛出std::bad_function_call异常。

示例:

#include <functional>
#include <iostream>void print_num(int i) { std::cout << i << '\n'; }int main() {// 存储自由函数std::function<void(int)> f_display = print_num;f_display(-9);// 存储 lambdastd::function<double(double)> f_dub = [](double a) { return 2.0 * a; };std::cout << f_dub(2) << '\n';// 存储目标为空std::function<int()> f = nullptr;try {f();} catch (const std::bad_function_call& e) { //抛出异常std::cout << e.what() << '\n';}
}

输出结果为:

-9
4
bad_function_call

在了解完function的用法之后,回到我们最开始的问题,其中,6次循环中要处理的目标的特征标均为double(double),因此,我们班使用function包装器将它们将统一“包装”成function<double(double)类型,这样模板函数use_f将只实例化一次。使用function包装器改进后的代码如下所示:

#include <functional>
#include <iostream>template <typename T, typename F>
T use_f(T v, F f) {static int count = 0;count++;std::cout << " use_f count = " << count << ", &count = " << &count<< std::endl;return f(v);
}class Fp {private:double z_;public:Fp(double z = 1.0) : z_(z) {}double operator()(double p) { return z_ * p; }
};class Fq {private:double z_;public:Fq(double z = 1.0) : z_(z) {}double operator()(double q) { return z_ + q; }
};double dub(double x) { return 2.0 * x; }
double square(double x) { return x * x; }
int main() {using std::cout;using std::endl;using std::function;double y = 1.21;typedef function<double(double)> fdd;  // simplify the type declarationcout << "Function pointer dub:\n";cout << " " << use_f(y, fdd(dub)) << endl;cout << "Function pointer square:\n";cout << " " << use_f(y, fdd(square)) << endl;cout << "Function object Fp:\n";cout << " " << use_f(y, fdd(Fq(10.0))) << endl;cout << "Function object Fq:\n";cout << " " << use_f(y, fdd(Fp(10.0))) << endl;cout << "Lambda expression 1:\n";cout << " " << use_f(y, fdd([](double u) { return u * u; })) << endl;cout << "Lambda expression 2:\n";cout << " " << use_f<double>(y, fdd([](double u) { return u + u / 2.0; }))<< endl;return 0;
}

输出结果为:

Function pointer dub:use_f count = 1, &count = 0x555555758134
2.42
Function pointer square:use_f count = 2, &count = 0x555555758134
1.4641
Function object Fp:use_f count = 3, &count = 0x555555758134
11.21
Function object Fq:use_f count = 4, &count = 0x555555758134
12.1
Lambda expression 1:use_f count = 5, &count = 0x555555758134
1.4641
Lambda expression 2:use_f count = 6, &count = 0x555555758134
1.815

从输出结果可以看出,use_f确实只实例化了一次,增加了编码效率,6次循环调用同一个函数,增加了代码额执行效率。

总结

function包装器将可调用对象的类型进行统一,便于我们对其进行统一化管理,同时,使用function包装器可以解决模板效率低下,实例化多份的问题。

参考文献

C++ Primer Plus(第六版) - 第18章 探讨C++新标准
C++ 参考手册

相关文章:

C++11 包装器function

文章首发公众号&#xff1a;iDoitnow C提供了多个包装器&#xff0c;它们主要是为了给其他编程接口提供更一致或更合适的接口。C11提供了多个包装器&#xff0c;这里我们重点了解一下包装器function。 对于function, C 参考手册给出的定义为&#xff1a; 类模板 std::function…...

XCP实战系列介绍14-基于Vector_Davinci工具的XCP配置介绍(三)

本文框架 1.概述2. 其他模块配置2.1 XCP初始化3. 手工代码部分3.1 周期函数添加3.2 DAQ Event调用3.3 XCP模块本身代码3.4 标定量的添加1.概述 在对XCP的配置部分介绍中我们计划分别对通讯部分配置、XCP模块本身配置及其他相关模块配置三篇进行介绍,在前两篇我们介绍了XCP配置…...

计算机图形学:中点BH算法对任意斜率的直线扫描转换方法

作者&#xff1a;非妃是公主 专栏&#xff1a;《计算机图形学》 博客地址&#xff1a;https://blog.csdn.net/myf_666 个性签&#xff1a;顺境不惰&#xff0c;逆境不馁&#xff0c;以心制境&#xff0c;万事可成。——曾国藩 文章目录专栏推荐专栏系列文章序一、问题提出二、…...

(十一)、用户中心页面【uniapp+uinicloud多用户社区博客实战项目(完整开发文档-从零到完整项目)】

1,个人中心页面 1.1 新建个人中心页面 1.2 纯净版个人中心页面代码&#xff1a; <template><view class"user"><view class"top"><view class"group"><view class"userinfo"><!-- 顶部 左侧 头像 …...

LA@复数和复矩阵@实对称阵相关定理

文章目录复数&#x1f388;复矩阵和复向量共轭矩阵性质定理实对称阵的相关定理复数&#x1f388; 复数 (数学) (wikipedia.org) 加法&#xff1a;(abi)(cdi)(ac)(bd)i)减法&#xff1a;(abi)−(cdi)(a−c)(b−d)i)乘法&#xff1a;(abi)(cdi)acbciadibdi2(ac−bd)(bcad)i除法&…...

cmd set命令笔记

使用 set是cmd最基础的命令&#xff0c;每个人都会用&#xff0c;但其实它还是有些知识的。 set 用来接收入参 set /p var请选择&#xff08;1或2或3&#xff09;: echo %var%可以接收输入的参数。 set /p var请选择&#xff08;1或2或3&#xff09;: echo %var% 语法 he…...

IB学校获得IBO授权究竟有多难?

IB 学校认证之路&#xff0c;道阻且长 The road to IB school accreditation is long and difficult一所学校能获得IB授权必须经过IBO非常严格的审核&#xff0c;在办学使命&教育理念、组织架构、师资力量&授课技能、学校硬件设施和课程体系上完全符合标准才可获得授权…...

火山引擎 DataTester:A/B 测试,让企业摆脱广告投放“乱烧钱”

更多技术交流、求职机会&#xff0c;欢迎关注字节跳动数据平台微信公众号&#xff0c;回复【1】进入官方交流群 在广告投放的场景下&#xff0c;一线广告优化师通常会创建多个计划&#xff0c;去测试不同的广告素材效果。这套方法看似科学&#xff0c;实际上却存在诸多问题&…...

黑马redis学习记录:缓存

一、介绍 什么是缓存&#xff1f; 缓存(Cache)&#xff0c;就是数据交换的缓冲区&#xff0c;俗称的缓存就是缓冲区内的数据&#xff0c;一般从数据库中获取&#xff0c;存储于本地代码 缓存无处不在 为什么要使用缓存&#xff1f; 因为速度快,好用缓存数据存储于代码中,而…...

CD20靶向药物|适应症|市场销售-上市药品前景分析

CD20是靶向治疗的第一个靶点&#xff0c;是B细胞淋巴瘤的现代治疗药物。CD20作为治疗剂的使用被认为是方便的&#xff0c;原因有二。首先&#xff0c;在 CD20 阳性肿瘤的情况下&#xff0c;这种受体大量存在于 B 淋巴细胞表面——每个细胞大约有十万个分子。其次&#xff0c;干…...

多源 复制

使复制从属服务器能够同时从多个主服务器接收事务至少需要两个主服务器和一个从属服务器设备从属服务器为每个主服务器创建一个 复制通道从属服务器必须使用基于表的资料档案库多源复制与基于文件的资料档案库不兼容不尝试检测或解决冲突如果需要此功能&#xff0c;则由应用程序…...

微服务项目【消息推送(RabbitMQ)】

创建消费者 第1步&#xff1a;基于Spring Initialzr方式创建zmall-rabbitmq消费者模块 第2步&#xff1a;在公共模块中添加rabbitmq相关依赖 <!--rabbitmq--> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-bo…...

vr电力刀闸事故应急演练实训系统开发

电力事故是在电力生产和输电过程中可能发生的意外事件&#xff0c;它们可能会对人们的生命财产安全造成严重的威胁。因此&#xff0c;电力事故应急演练显得尤为重要。而VR技术则可以为电力事故应急演练提供一种全新的解决方案。 在虚拟环境中&#xff0c;元宇宙VR会模拟各种触电…...

C++类和对象补充

目录 前言&#xff1a; 1. 构造函数->初始化列表 1.1 初始化列表出现原因 1.2 初始化列表写法 2. explicit关键字 2.1 explict的出现 2.2 explict的写法 3. static成员 4. 友元 4.1 友元函数 4.2 友元类 5. 内部类和匿名对象 5.1 内部类 5.2 匿名对象 前言&a…...

08 SpringCloud 微服务网关Gateway组件

网关简介 大家都都知道在微服务架构中&#xff0c;一个系统会被拆分为很多个微服务。那么作为客户端要如何去调用这么多的微服务呢&#xff1f; 如果没有网关的存在&#xff0c;我们只能在客户端记录每个微服务的地址&#xff0c;然后分别去用。 这样的架构&#xff0c;会存…...

极验3代 加密分析

目标链接 aHR0cHM6Ly93d3cuZ2VldGVzdC5jb20vZGVtby9zbGlkZS1mbG9hdC5odG1s接口分析 极验参数重要信息 gt和challenge&#xff1b;gt是固定的&#xff0c;但是challenge每次请求会产生不同的&#xff0c;这里的请求的并没有什么加密参数。 下一个请求 gettype.php&#xff0c…...

python 数据分析可视化实战 超全 附完整代码数据

代码数据&#xff1a;https://download.csdn.net/download/qq_38735017/873799141.1 数据预处理1.1.1 异常值检测①将支付时间转为标准时间的过程中发生错误&#xff0c;经排查错误数据为‘2017/2/29’,后将其修改为‘2017/2/27’。②经检测发现部分订单应付金额与实付金额都为…...

有趣的HTML实例(十三) 咖啡选择(css+js)

一个人追求目标的路途是孤单的&#xff0c;一个人独品辛酸的时候是寂寥的&#xff0c;一个人马不停蹄的追赶着&#xff0c;狂奔着&#xff0c;相信前方是一片光明&#xff0c;我从不放弃希望&#xff0c;就像我对生活的信念&#xff0c;没有人可以动摇。 ——《北京青年》 目录…...

【力扣-LeetCode】1139. 最大的以 1 为边界的正方形 C++题解

1139. 最大的以 1 为边界的正方形难度中等137收藏分享切换为英文接收动态反馈给你一个由若干 0 和 1 组成的二维网格 grid&#xff0c;请你找出边界全部由 1 组成的最大 正方形 子网格&#xff0c;并返回该子网格中的元素数量。如果不存在&#xff0c;则返回 0。示例 1&#xf…...

【JavaGuide面试总结】Redis篇·下

【JavaGuide面试总结】Redis篇下1.如何使用 Redis 事务&#xff1f;2.如何解决 Redis 事务的缺陷&#xff1f;3.说说Redis bigkey吧4.大量 key 集中过期问题怎么解决的5.如何保证缓存和数据库数据的一致性&#xff1f;6.缓存穿透有哪些解决办法&#xff1f;7.缓存击穿有哪些解决…...

ForkJoinPool原理

1、概述 Fork/Join框架是Java7提供了的一个用于并行执行任务的框架。ForkJoinPool是Java中提供了一个线程池&#xff0c;特点是用来执行分治任务。主题思想是将大任务分解为小任务&#xff0c;然后继续将小任务分解&#xff0c;直至能够直接解决为止&#xff0c;然后再依次将任…...

02 python基本语法和数据类型

基本语法 python脚本可以在python交互式shell或者代码编辑器中编写与运行。python文件的扩展名一般为.py python使用缩进来区分不同的代码块,此特性有利于提高代码可读性。 下面是一个简单的python条件语句代码: 小明=矮穷错 小红=白富美 小华=高富帅 小李=程序员某人 = &quo…...

【办公类-16-09】“2022下学期 大班运动场地分配表-跳过节日循环排序”(python 排班表系列)

样例展示&#xff1a;跳过节日的运动场地循环排序表&#xff08;8个班级8组内容 下学期一共20周&#xff09;背景需求&#xff1a;上学期做过一次大班运动场地安排&#xff0c;跳过节日。2023.2下学期运动场地排班&#xff08;跳过节日&#xff09;又来了。一、场地器械微调二、…...

全网多种方法分析解决HTTP Status 404资源未找到的错误,TCP的3次握手,dns域名解析,发起http请求以及cookie和session的区别

文章目录1. 文章引言2. 简述URL3. http完整请求3.1 DNS域名解析3.2 TCP的3次握手3.3 发起http请求3.4 浏览器解析html代码3.5 浏览器对页面进行渲染呈现给用户4. 解决404错误的方法5. 补充知识点5.1 cookie和session的区别5.2 ChatGPT的介绍1. 文章引言 正赶上最近ChatGPT很火…...

Django图书商场购物系统python毕业设计项目推荐

mysql数据库进行开发&#xff0c;实现了首页、个人中心、用户管理、卖家管理、图书类型管理、图书信息管理、订单管理、系统管理等内容进行管理&#xff0c;本系统具有良好的兼容性和适应性&#xff0c;为用户提供更多的网上图书商城信息&#xff0c;也提供了良好的平台&#x…...

基于模型预测控制(MPC)的悬架系统仿真分析

目录 前言 1.悬架系统 2.基于MPC的悬架系统仿真分析 2.1 simulink模型 2.2仿真结果 2.3 结论 3 总结 前言 模型预测控制是无人驾驶中较为热门的控制算法&#xff0c;但是对于悬架等这类系统的控制同样适用。 我们知道模型预测控制主要可以划分为三个部分&#xff1a; …...

Word处理控件Aspose.Words功能演示:使用 Java 拆分 MS Word 文档

Aspose.Words 是一种高级Word文档处理API&#xff0c;用于执行各种文档管理和操作任务。API支持生成&#xff0c;修改&#xff0c;转换&#xff0c;呈现和打印文档&#xff0c;而无需在跨平台应用程序中直接使用Microsoft Word。此外&#xff0c;API支持所有流行的Word处理文件…...

图扑数字孪生智慧机场,助推民航“四型机场“建设

前言 民航局印发的《智慧民航建设路线图》文件中&#xff0c;明确提出智慧机场是智慧民航的四个核心抓手之一。并从机场全域协同运行、作业与服务智能化、智慧建造与运维方面&#xff0c;为智慧机场的发展绘制了清晰的蓝图。 效果展示 图扑软件应用自主研发核心产品 HT for …...

内网安装管家婆软件如何实现外网访问?内网穿透的几种方案教程

管家婆软件从网络架构上分两种版本&#xff1a;web&#xff08;浏览器http端口&#xff09;访问的版本和客户端&#xff08;211固定端口sqlserver数据库&#xff09;访问的版本。公司库管经常用仓库登录管家婆&#xff0c;一旦需要在公司外部登陆访问管家婆客户端&#xff0c;就…...

CCNP350-401学习笔记(1-50题)

1、Which function does a fabric edge node perform in an SD-Access deployment?A. Connects endpoints to the fabric and forwards their traffic. B. Encapsulates end-user data traffic into LISP. C. Connects the SD-Access fabric to another fabric or external La…...

深圳建网站制作维护/竞价排名的定义

1. 单线程 js是单线程语言 其异步执行其实是通过事件循环机制模拟出来的&#xff0c;而不是真正的开辟新的物理线程。 为什么js是单线程呢 浏览页面是应用需求&#xff0c;没有很高的实时性需求。js设计为单线程避免了页面交互时因线程执行顺序的不确定给页面渲染带来的不确定…...

wordpress淘宝客模板修改教程/谷歌广告优化

SQL分类 DDL&#xff08;Data Definition Languages&#xff09; 数据定义语言&#xff0c;这些语句定义了不同的数据字段、数据库、表、列、索引等数据。 常用的语句关键字主要包括 create(添加)&#xff0c;drop&#xff08;删除&#xff09;&#xff0c;alter&#xff08;修…...

苏州建站公司兴田德润i网址多少/大数据培训班出来能就业吗

(声明:本系列所用的模式都来自GOF23中&#xff0c;本系列并不是讲23种经典设计模式&#xff0c;而是如何去使用这些模式) 前面我们设计了员工的工资&#xff0c;奖金&#xff0c;福利系统。今天客户又来增加需求了:"我们需要统计公司&#xff0c;部门&#xff0c;或者小…...

网站建设需要用到什么/sem广告投放是做什么的

React团队推出了一款新工具&#xff0c;希望帮助开发人员减轻新建React应用所引发的痛苦。 在一篇博文中&#xff0c;Dan Abramov介绍了Create React App。该工具让开发人员可以使用一行命令新建一个React应用程序——包括其构建过程和依赖。这是官方支持的一种React应用程序创…...

白城网站建设哪家好/衡阳有实力seo优化

算法概述 在失真图像的基础上产生了一个幻觉参考图像&#xff0c;将失真图像和幻觉参考图像成对输入回归网络中&#xff0c;同时结合生成网络中提取的部分信息&#xff0c;最终产生图像的质量分数。 创新点 首先在回归网络中&#xff0c;该网络将感知差异信息融合入回归网络…...

建设独立服务器网站/百度关键词竞价查询系统

“Java和C最大的不同在于Java采用的指针模型可以消除重写内存和损坏数据的可能性。” “Java编译器能够检测许多在其他语言中仅在运行时才能够检测出来的问题。”...