[C++] C++ 11的functional模块介绍和使用案例
functional模块介绍
functional模块是C++ 11提供了一组函数对象和算法,用于增强C++的函数式编程能力。该模块中的函数对象和算法可以大大简化代码,并提供了一些有用的工具,例如函数适配器和函数对象的组合。
functional模块中的函数对象包括:
- plus:加法函数对象,用于将两个参数相加。
- minus:减法函数对象,用于将第一个参数减去第二个参数。
- multiplies:乘法函数对象,用于将两个参数相乘。
- divides:除法函数对象,用于将第一个参数除以第二个参数。
- modulus:取模函数对象,用于返回第一个参数除以第二个参数的余数。
- negate:取反函数对象,将参数取反。
functional模块中的算法包括:
- bind:将一个函数对象和一些参数绑定在一起,生成一个新的函数对象。
- mem_fn:将一个成员函数包装成函数对象。
- not1:对一个函数对象进行逻辑非运算。
- ref:将一个对象包装成一个引用包装器。
- cref:将一个对象包装成一个const引用包装器。
使用案例
案例一
#include <iostream>
#include <functional>int main() {std::plus<int> plusFunc;int result = plusFunc(1, 2); // 将1和2相加,返回3std::cout << result << std::endl;return 0;
}
在上面的例子中,我们创建了一个std::plus<int>的实例plusFunc,该实例可以用于将两个整数相加。然后,我们调用plusFunc函数对象,传入参数1和2,并将返回的结果赋值给result变量。最后,我们将结果输出到控制台,得到的输出结果是3。
使用std::plus函数对象的好处是,它可以与其他函数对象和算法结合使用,例如使用std::transform算法对一个容器中的元素进行相加操作:
#include <iostream>
#include <functional>
#include <vector>
#include <algorithm>int main() {std::vector<int> nums {1, 2, 3, 4, 5};std::vector<int> result(nums.size());std::transform(nums.begin(), nums.end(), nums.begin(), std::negate<int>());std::transform(nums.begin(), nums.end(), result.begin(), std::bind(std::plus<int>(), std::placeholders::_1, 2));for (int num : result) {std::cout << num << " ";}return 0;
}
在上面的例子中,我们首先使用std::transform算法和std::negate函数对象对nums容器中的元素进行取反操作。然后,使用std::transform算法和std::bind函数将nums中的每个元素与2相加,并将结果存储在result容器中。最后,我们遍历result容器并输出每个元素的值。输出结果为-1 0 1 2 3,表示nums中的每个元素都加上了2。
这个例子展示了std::plus函数对象的灵活性,它可以与其他函数对象和算法结合使用,简化了代码,并提供了更好的可读性和可维护性。
案例二
#include <iostream>
#include <functional>int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}int main() {std::function<int(int, int)> func1 = add;std::function<int(int, int)> func2 = std::bind(subtract, std::placeholders::_2, std::placeholders::_1);int result1 = func1(1, 2); // 调用add函数,返回3int result2 = func2(2, 1); // 调用subtract函数,返回1std::cout << result1 << std::endl;std::cout << result2 << std::endl;return 0;
}
在上面的例子中,我们定义了两个函数add和subtract,并使用std::function将它们包装成函数对象func1和func2。然后,我们可以像调用普通函数一样使用这些函数对象,通过调用func1和func2来执行对应的函数。输出结果分别为3和1。
在这个例子中,我们使用了bind函数将subtract函数的两个参数绑定在一起,然后通过调整参数的顺序来实现减法运算。这是functional模块中的一个常见用法,可以用来生成新的函数对象。
案例三
#include <iostream>
#include <functional>class MyClass {
public:void printMessage(const std::string& message) {std::cout << message << std::endl;}
};int main() {MyClass obj;auto printFunc = std::mem_fn(&MyClass::printMessage);printFunc(obj, "Hello, world!"); // 调用obj对象的printMessage函数,并传入参数"Hello, world!"return 0;
}
在上面的例子中,我们定义了一个名为MyClass的类,该类具有一个成员函数printMessage,用于输出一条消息。然后,我们创建了一个MyClass类型的对象obj,并使用std::mem_fn将printMessage函数转化为一个函数对象printFunc。最后,我们调用printFunc函数对象,传入obj对象和字符串参数"Hello, world!",从而调用了obj对象的printMessage函数并输出了消息。
使用std::mem_fn函数对象的好处是,它允许我们以通用的方式处理成员函数指针和成员函数对象,可以与其他函数对象和算法结合使用。例如,我们可以通过std::transform算法将一个容器中的每个元素调用成员函数并获取结果:
#include <iostream>
#include <functional>
#include <vector>
#include <algorithm>class MyClass {
public:int multiply(int x, int y) {return x * y;}
};int main() {std::vector<int> nums {1, 2, 3, 4, 5};std::vector<int> result(nums.size());MyClass obj;auto multiplyFunc = std::mem_fn(&MyClass::multiply);std::transform(nums.begin(), nums.end(), result.begin(), std::bind(multiplyFunc, &obj, std::placeholders::_1, 2));for (int num : result) {std::cout << num << " ";}return 0;
}
在上面的例子中,我们定义了一个名为MyClass的类,该类具有一个成员函数multiply,用于计算两个整数的乘积。然后,我们创建了一个MyClass类型的对象obj,并使用std::mem_fn将multiply函数转化为一个函数对象multiplyFunc。最后,我们使用std::transform算法和std::bind函数将nums容器中的每个元素和数字2传入multiplyFunc函数对象中进行计算,并将结果存储在result容器中。最后,我们遍历result容器并输出每个元素的值。输出结果为2 4 6 8 10,表示nums中的每个元素都乘以了2。
这个例子展示了std::mem_fn函数对象的灵活性,它允许我们处理成员函数指针和成员函数对象,并与其他函数对象和算法结合使用,简化了代码,并提供了更好的可读性和可维护性。
案例四
#include <iostream>
#include <functional>void updateValue(int& value) {value += 10;
}int main() {int num = 5;std::cout << "Before update: " << num << std::endl;std::function<void()> updateFunc = std::bind(updateValue, std::ref(num));updateFunc();std::cout << "After update: " << num << std::endl;return 0;
}
在上面的例子中,我们定义了一个名为updateValue的函数,该函数接受一个引用参数,并将其值增加10。然后,我们创建了一个名为num的整数变量,初始值为5。接下来,我们使用std::ref函数模板将num对象转化为一个引用包装器,并通过std::bind函数将updateValue函数与引用包装器绑定到一起,创建一个函数对象updateFunc。最后,我们调用updateFunc函数对象,从而实际调用了updateValue函数,并传递了num对象的引用作为参数。输出结果为:
Before update: 5
After update: 15
可以看到,通过使用std::ref函数模板,我们可以在函数调用中传递对象的引用,从而实现对对象的修改。
案例五
#include <iostream>
#include <functional>void printValue(const int& value) {std::cout << "Value: " << value << std::endl;
}int main() {int num = 10;std::function<void()> printFunc = std::bind(printValue, std::cref(num));printFunc();return 0;
}
在上面的例子中,我们定义了一个名为printValue的函数,该函数接受一个常量引用参数,并打印该值。然后,我们创建了一个名为num的整数变量,初始值为10。接下来,我们使用std::cref函数模板将num对象转化为一个常量引用包装器,并通过std::bind函数将printValue函数与常量引用包装器绑定到一起,创建一个函数对象printFunc。最后,我们调用printFunc函数对象,从而实际调用了printValue函数,并传递了num对象的常量引用作为参数。输出结果为:
Value: 10
可以看到,通过使用std::cref函数模板,我们可以在函数调用中传递对象的常量引用,从而实现对对象的只读访问。
案例六
#include <iostream>
#include <functional>
#include <vector>
#include <algorithm>bool isOdd(int num) {return num % 2 != 0;
}int main() {std::vector<int> nums {1, 2, 3, 4, 5};std::vector<int> evenNums;std::copy_if(nums.begin(), nums.end(), std::back_inserter(evenNums), std::not1(std::ptr_fun(isOdd)));for (int num : evenNums) {std::cout << num << " ";}std::cout << std::endl;return 0;
}
在上面的例子中,我们定义了一个名为isOdd的谓词函数,该函数接受一个整数参数,并判断该整数是否为奇数。然后,我们创建了一个整数类型的vector容器nums,并使用std::copy_if算法和std::not1函数模板将nums容器中的所有偶数元素复制到另一个vector容器evenNums中。在调用std::copy_if算法时,我们使用std::not1函数模板将isOdd谓词函数进行否定操作,从而实现对偶数元素的筛选。最后,我们遍历evenNums容器并输出其中的所有元素。输出结果为:
2 4
这个例子展示了std::not1函数模板的用途,可以对谓词函数进行否定操作,从而实现相反的逻辑判断。在实际开发中,std::not1函数模板可以用于各种需要对谓词函数进行逻辑反操作的场景,例如在STL算法中对元素进行筛选、排序或处理等操作。
相关文章:
[C++] C++ 11的functional模块介绍和使用案例
functional模块介绍 functional模块是C 11提供了一组函数对象和算法,用于增强C的函数式编程能力。该模块中的函数对象和算法可以大大简化代码,并提供了一些有用的工具,例如函数适配器和函数对象的组合。 functional模块中的函数对象包括&am…...

kubernetes基本概念和操作
基本概念和操作 1.Namespace1.1概述1.2应用示例 2.Pod2.1概述2.2语法及应用示例 3.Label3.1概述3.2语法及应用示例 4.Deployment4.1概述4.2语法及应用示例 5.Service5.1概述5.2语法及应用示例5.2.1创建集群内部可访问的Service5.2.2创建集群外部可访问的Service5.2.3删除服务5.…...
20240128周报-网络太杂,Tomcat太难
今天来做个小总结吧,之前说想用几个月的时间将Java生态给整理一遍,该工作已经进入第三周了。先和各位老老板汇报一下上一周的工作,然后说一下本周的计划和后面的计划。 1.上周工作 上周的计划是将网络和Tomcat的内容梳理一番,但…...

DES加密原理
DES加密算法综合运用了置换、代替、代数等多种密码技术,具有设计精 巧、实现容易、使用方便等特点。DES加密算法的明文、密文和密钥的分组长度 都是64位,详细的DES加密算法结构如图6-10所示。 图6-10 DES加密算法结构图 DES加密过程如下所示ÿ…...

react 之 useCallback
简单讲述下useCallback的使用方法,useCallback也是用来缓存的,只不过是用于做函数缓存 // useCallbackimport { memo, useCallback, useState } from "react"const Input memo(function Input ({ onChange }) {console.log(子组件重新渲染了…...

OfficeWeb365 Readfile 任意文件读取漏洞复现
0x01 产品简介 OfficeWeb365 是专注于 Office 文档在线预览及PDF文档在线预览云服务,包括 Microsoft Word 文档在线预览、Excel 表格在线预览、Powerpoint 演示文档在线预览,WPS 文字处理、WPS 表格、WPS 演示及 Adobe PDF 文档在线预览。 0x02 漏洞概述 OfficeWeb365 Rea…...
UnityShader(十三)Unity内置的函数
在计算光照模型时我们需要得到许多数据,比如光源方向、视角方向这种基本信息。 在之前的例子中都是自行在代码里计算的,比如: normalize(_WorldSpaceLight0Pos.xyz) 得到光源方向(这种方法实际只适用平行光) normaliz…...

【开源】基于Qt5的ROS1/ROS2人机交互软件(支持地图编辑/多点导航)
本项目基于Qt5开发,基于CMake进行构建,可以实现一套代码同时在ROS1/ROS2系统中使用(本项目已接入CI,保证多ROS版本/系统版本可用性) 项目地址: https://github.com/chengyangkj/Ros_Qt5_Gui_App 软件在编译时会自动识别环境变量中的ROS1/ROS…...
Spring和SpringBoot的区别是什么
Spring 和 Spring Boot 是 Java 开发领域内两个极其重要且紧密相关的框架,它们各自在企业级应用开发中扮演着不同的角色,并带来了一系列革新性的变化。以下是关于两者之间主要区别的详细分析: 一、设计理念与定位 Spring Framework Spring 是…...

布局技巧及CSS初始化
一,margin负值巧妙应用 二,文字围绕浮动元素 三,行内块 四,CSS三角强化 五,CSS初始化 一,margin负值巧妙应用 制作盒子的细线边框: 鼠标经过li后变色: 二,文字围绕…...

excel怎么设置密码?轻松保护您的工作表
在数字化时代,数据的安全性显得尤为重要。excel作为我们日常工作中广泛使用的办公软件,其中可能包含了大量的敏感数据。为了确保这些数据不被未授权的人访问,本文将为您详细介绍excel怎么设置密码,从而有效地保护您的数据安全。 方…...

鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之TimePicker组件
鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之TimePicker组件 一、操作环境 操作系统: Windows 10 专业版、IDE:DevEco Studio 3.1、SDK:HarmonyOS 3.1 二、TimePicker组件 TextClock组件通过文本将当前系统时间显示在设备上。支持不…...

Springboot集成Camunda并完成一条流程实例
💖专栏简介 ✔️本专栏将从Camunda(卡蒙达) 7中的关键概念到实现中国式工作流相关功能。 ✔️文章中只包含演示核心代码及测试数据,完整代码可查看作者的开源项目snail-camunda ✔️请给snail-camunda 点颗星吧😘 💖设计流程定…...

宠物用品/宠物自动饮水机方案
宠物自动饮水机方案原理 宠物自动饮水机,也叫做智能宠物饮水机,是一种为宠物设计的智能化饮水器。应用核心主要在于智能化水泵控制,以及外围传感器电路。 宠物自动饮水机使用方便,不用频繁的换水。另外,自来水的水质可…...

git小白进阶之路
git是最常用的版本控制工具,我对其进行了整理后续补充,这个文档欢迎大家来讨论,当前我的视频梳理: git小白进阶之路_哔哩哔哩_bilibili,非常希望大佬们能够批评指正,并多多交流。 目录 初始配置 配置账号…...
哈希表——C++
目录 一、首先使用拉链法: 二、开放寻址法 三、字符串哈希 1.具体如何使用进制的方式来存储字符前缀的可以看这个y总的这个图 2.接下来说一说算某个中间的区间的字符串哈希值 哈希表是一种数组之间互相映射的数据结构,比如举个简单的例子一个十个的数…...

LabVIEW叶片厚度远程监控
LabVIEW叶片厚度远程监控 随着网络技术的高速发展,远程监控广泛应用在各个领域。本文介绍了一种基于LabVIEW的植物叶片厚度远程监控系统,旨在实现对植物生长状况的精准监测和分析。 该系统利用LabVIEW软件开发工具,通过TCP网络协议实现数据…...

el-table动态合并
废话就不多说了,直接上代码!!! 合并行 // 方法一 <template><div class"container"><el-table :data"dataSource" :border"true":header-cell-style"{ font-weight: normal,…...

【DevOps】产品需求文档(PRD)与常见原型软件
文章目录 1、PRD介绍1.1、概述1.2、前提条件1.3、主要目的1.4、关键内容1.5、表述方式1.6、需求评审人员1.7、一般内容结构 2、需求流程3、常见原型软件3.1、Word3.2、Axure3.2.1、详细介绍3.2.2、应用分类3.2.3、优缺点 3.3、摹客RP3.4、蓝湖3.5、GUI Design Studio 1、PRD介绍…...
【QT+QGIS跨平台编译】之十八:【Expat+Qt跨平台编译】(一套代码、一套框架,跨平台编译)
文章目录 一、Expat介绍二、文件下载三、文件分析四、pro文件五、编译实践一、Expat介绍 Expat库最初由James Clark创建,已经成为许多编程语言中常用的XML解析工具。它以其简单、快速和可靠的特点而受到广泛的认可和使用。 Expat库的优点包括: 快速:Expat的解析速度非常快…...

Flask RESTful 示例
目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题: 下面创建一个简单的Flask RESTful API示例。首先,我们需要创建环境,安装必要的依赖,然后…...

Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...

Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...

dedecms 织梦自定义表单留言增加ajax验证码功能
增加ajax功能模块,用户不点击提交按钮,只要输入框失去焦点,就会提前提示验证码是否正确。 一,模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...

高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
什么是EULA和DPA
文章目录 EULA(End User License Agreement)DPA(Data Protection Agreement)一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA(End User License Agreement) 定义: EULA即…...
HTML前端开发:JavaScript 常用事件详解
作为前端开发的核心,JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例: 1. onclick - 点击事件 当元素被单击时触发(左键点击) button.onclick function() {alert("按钮被点击了!&…...

【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...

k8s业务程序联调工具-KtConnect
概述 原理 工具作用是建立了一个从本地到集群的单向VPN,根据VPN原理,打通两个内网必然需要借助一个公共中继节点,ktconnect工具巧妙的利用k8s原生的portforward能力,简化了建立连接的过程,apiserver间接起到了中继节…...
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
以下是一个完整的 Angular 微前端示例,其中使用的是 Module Federation 和 npx-build-plus 实现了主应用(Shell)与子应用(Remote)的集成。 🛠️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...