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

[C++] C++ 11的functional模块介绍和使用案例

functional模块介绍

functional模块是C++ 11提供了一组函数对象和算法,用于增强C++的函数式编程能力。该模块中的函数对象和算法可以大大简化代码,并提供了一些有用的工具,例如函数适配器和函数对象的组合。

functional模块中的函数对象包括:

  1. plus:加法函数对象,用于将两个参数相加。
  2. minus:减法函数对象,用于将第一个参数减去第二个参数。
  3. multiplies:乘法函数对象,用于将两个参数相乘。
  4. divides:除法函数对象,用于将第一个参数除以第二个参数。
  5. modulus:取模函数对象,用于返回第一个参数除以第二个参数的余数。
  6. negate:取反函数对象,将参数取反。

functional模块中的算法包括:

  1. bind:将一个函数对象和一些参数绑定在一起,生成一个新的函数对象。
  2. mem_fn:将一个成员函数包装成函数对象。
  3. not1:对一个函数对象进行逻辑非运算。
  4. ref:将一个对象包装成一个引用包装器。
  5. 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提供了一组函数对象和算法&#xff0c;用于增强C的函数式编程能力。该模块中的函数对象和算法可以大大简化代码&#xff0c;并提供了一些有用的工具&#xff0c;例如函数适配器和函数对象的组合。 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太难

今天来做个小总结吧&#xff0c;之前说想用几个月的时间将Java生态给整理一遍&#xff0c;该工作已经进入第三周了。先和各位老老板汇报一下上一周的工作&#xff0c;然后说一下本周的计划和后面的计划。 1.上周工作 上周的计划是将网络和Tomcat的内容梳理一番&#xff0c;但…...

DES加密原理

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

react 之 useCallback

简单讲述下useCallback的使用方法&#xff0c;useCallback也是用来缓存的&#xff0c;只不过是用于做函数缓存 // 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内置的函数

在计算光照模型时我们需要得到许多数据&#xff0c;比如光源方向、视角方向这种基本信息。 在之前的例子中都是自行在代码里计算的&#xff0c;比如&#xff1a; normalize(_WorldSpaceLight0Pos.xyz) 得到光源方向&#xff08;这种方法实际只适用平行光&#xff09; normaliz…...

【开源】基于Qt5的ROS1/ROS2人机交互软件(支持地图编辑/多点导航)

本项目基于Qt5开发&#xff0c;基于CMake进行构建&#xff0c;可以实现一套代码同时在ROS1/ROS2系统中使用(本项目已接入CI,保证多ROS版本/系统版本可用性) 项目地址&#xff1a; https://github.com/chengyangkj/Ros_Qt5_Gui_App 软件在编译时会自动识别环境变量中的ROS1/ROS…...

Spring和SpringBoot的区别是什么

Spring 和 Spring Boot 是 Java 开发领域内两个极其重要且紧密相关的框架&#xff0c;它们各自在企业级应用开发中扮演着不同的角色&#xff0c;并带来了一系列革新性的变化。以下是关于两者之间主要区别的详细分析&#xff1a; 一、设计理念与定位 Spring Framework Spring 是…...

布局技巧及CSS初始化

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

excel怎么设置密码?轻松保护您的工作表

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

鸿蒙(HarmonyOS)项目方舟框架(ArkUI)之TimePicker组件

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

Springboot集成Camunda并完成一条流程实例

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

宠物用品/宠物自动饮水机方案

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

git小白进阶之路

git是最常用的版本控制工具&#xff0c;我对其进行了整理后续补充&#xff0c;这个文档欢迎大家来讨论&#xff0c;当前我的视频梳理&#xff1a; git小白进阶之路_哔哩哔哩_bilibili&#xff0c;非常希望大佬们能够批评指正&#xff0c;并多多交流。 目录 初始配置 配置账号…...

哈希表——C++

目录 一、首先使用拉链法&#xff1a; 二、开放寻址法 三、字符串哈希 1.具体如何使用进制的方式来存储字符前缀的可以看这个y总的这个图 2.接下来说一说算某个中间的区间的字符串哈希值 哈希表是一种数组之间互相映射的数据结构&#xff0c;比如举个简单的例子一个十个的数…...

LabVIEW叶片厚度远程监控

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

el-table动态合并

废话就不多说了&#xff0c;直接上代码&#xff01;&#xff01;&#xff01; 合并行 // 方法一 <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使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题&#xff1a; 下面创建一个简单的Flask RESTful API示例。首先&#xff0c;我们需要创建环境&#xff0c;安装必要的依赖&#xff0c;然后…...

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 发行版&#xff…...

dedecms 织梦自定义表单留言增加ajax验证码功能

增加ajax功能模块&#xff0c;用户不点击提交按钮&#xff0c;只要输入框失去焦点&#xff0c;就会提前提示验证码是否正确。 一&#xff0c;模板上增加验证码 <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&#xff08;End User License Agreement&#xff09;DPA&#xff08;Data Protection Agreement&#xff09;一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA&#xff08;End User License Agreement&#xff09; 定义&#xff1a; EULA即…...

HTML前端开发:JavaScript 常用事件详解

作为前端开发的核心&#xff0c;JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例&#xff1a; 1. onclick - 点击事件 当元素被单击时触发&#xff08;左键点击&#xff09; button.onclick function() {alert("按钮被点击了&#xff01;&…...

【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)

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

k8s业务程序联调工具-KtConnect

概述 原理 工具作用是建立了一个从本地到集群的单向VPN&#xff0c;根据VPN原理&#xff0c;打通两个内网必然需要借助一个公共中继节点&#xff0c;ktconnect工具巧妙的利用k8s原生的portforward能力&#xff0c;简化了建立连接的过程&#xff0c;apiserver间接起到了中继节…...

Angular微前端架构:Module Federation + ngx-build-plus (Webpack)

以下是一个完整的 Angular 微前端示例&#xff0c;其中使用的是 Module Federation 和 npx-build-plus 实现了主应用&#xff08;Shell&#xff09;与子应用&#xff08;Remote&#xff09;的集成。 &#x1f6e0;️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...