【STL十六】函数对象:包装器(std::function)——绑定器(std::bind)——函数适配器

news2024/10/2 6:32:42

【STL十六】函数对象:包装器(std::function)——绑定器(std::bind)——函数适配器

  • 一、包装器(std::function)
    • 1、简介
    • 2、头文件
    • 3、构造函数
    • 4、demo
    • 5、异常
  • 二、绑定器(std::bind)
    • 1、简介
    • 2、头文件
    • 3、构造函数
    • 4、demo:简单场景
    • 5、demo:值传递、引用传递
    • 6、demo:绑定6种可调用对象
  • 三、函数适配器
    • 1、函数适配器(function adapters)
    • 2、很多函数适配器在c++17就被废弃了
    • 3、bind就是一种函数适配器(绑定器)

一、包装器(std::function)

1、简介

  • std::function模板类是一个通用的可调用对象的包装器,
  • 作用:简单的、统一的方式处理可调用对象。

std::function 的实例能存储、复制及调用任何可调用 (Callable) 目标——函数、 lambda 表达式、 bind 表达式或其他函数对象,还有指向成员函数指针和指向数据成员指针。存储的可调用对象被称为 std::function 的目标。
若 std::function 不含目标,则称它为空。调用空 std::function 的目标导致抛出 std::bad_function_call 异常。

2、头文件

#include <functional>

3、构造函数

template< class >
class function; 

4、demo

#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "普通函数" << bh << "," << message << endl;
}

struct AA	// 类中有静态成员函数。
{
	static void show(int bh, const string& message) {
		cout << "成员函数" << bh << "," << message << endl;
	}
};

struct BB	// 仿函数。
{
	void operator()(int bh, const string& message) {
		cout << "仿函数 " << bh << "," << message << endl;
	}
};

struct CC	// 类中有普通成员函数。
{
	void show(int bh, const string& message) {
		cout << "成员函数" << bh << "," << message << endl;
	}
};
 
struct DD		                                 // 可以被转换为普通函数指针的类。
{
	using Fun = void (*)(int, const string&);    // 函数指针的别名。
	operator Fun() {
		return show;	                         // 返回普通函数show的地址。
	}
};

int main()
{
	using Fun = void(int, const string&);             // 函数类型的别名。

	// 普通函数。
	void(*fp1)(int, const string&) = show;	          // 声明函数指针,指向函数对象。
	fp1(1, "大唐不夜城。");						      // 用函数指针调用普通函数。
	function<void(int, const string&)> fn1 = show;    // 包装普通全局函数show。等价于 function<Fun> fn1 = show;
	fn1(1, "大唐不夜城。");							// 用function对象调用普通全局函数show。

	// 类的静态成员函数。
	void(*fp3)(int, const string&) = AA::show;	        // 用函数指针指向类的静态成员函数。
	fp3(2, "古城濮阳。");							    // 用函数指针调用类的静态成员函数。
	function<void(int, const string&)> fn3 = AA::show;	// 包装类的静态成员函数。
	fn3(2, "古城濮阳。");							    // 用function对象调用类的静态成员函数。

	// 仿函数。
	BB bb;
	bb(3, "古城西安。");		                        // 用仿函数对象调用仿函数。
	function<void(int, const string&)> fn4 = BB();		// 包装仿函数。
	fn4(3, "古城西安。");							    // 用function对象调用仿函数。

	// 创建lambda对象。
	auto lb = [](int bh, const string& message) {
		cout << "lambda表达式" << bh << "," << message << endl;
	};
	lb(4, "古城咸阳。");                                   // 调用lambda函数。
	function<void(int, const string&)> fn5 = lb;		   // 包装lamba函数。
	fn5(4, "古城咸阳。");								   // 用function对象调用lamba函数。

	// 类的非静态成员函数。
	CC cc;
	void (CC:: * fp11)(int, const string&) = &CC::show;		    // 定义类成员函数的指针。
	(cc.*fp11)(5, "咸阳秦始皇。");								// 用类成员函数的指针调用类的成员函数。
	function<void(CC&, int, const string&)> fn11 = &CC::show;	// 包装成员函数。
	fn11(cc, 5, "咸阳秦始皇。");						  		   // 用function对象调用成员函数。

	// 可以被转换为函数指针的类对象。
	DD dd;
	dd(6, "钟楼。");						            // 用可以被转换为函数指针的类对象调用普通函数。
	function<void(int, const string&)> fn12 = dd;	// 包装可以被转换为函数指针的类。
	fn12(6, "钟楼。");								// 用function对象调用它。
}

输出

普通函数1,大唐不夜城。
普通函数1,大唐不夜城。
成员函数2,古城濮阳。
成员函数2,古城濮阳。
仿函数 3,古城西安。
仿函数 3,古城西安。
lambda表达式4,古城咸阳。
lambda表达式4,古城咸阳。
成员函数5,咸阳秦始皇。
成员函数5,咸阳秦始皇。
普通函数6,钟楼。
普通函数6,钟楼。

5、异常

  • std::function对象未包装可调用对象,使用std::function对象将抛出std::bad_function_call异常
  • demo
#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "普通函数" << bh << "," << message << endl;
}
 
struct DD		                                 // 可以被转换为普通函数指针的类。
{
	using Fun = void (*)(int, const string&);    // 函数指针的别名。
	operator Fun() {
		return show;	                         // 返回普通函数show的地址。
	}
};

int main()
{
	DD dd;

	function<void(int, const string&)> fx /*= dd*/;
	try {  
		 fx(1, "xxxxxxxxxxxxxx。");
	}
	catch (std::bad_function_call e) {
		cout << "抛出了std::bad_function_call异常。";
	}
}

输出

抛出了std::bad_function_call异常。

二、绑定器(std::bind)

1、简介

  • std::bind()模板函数是一个通用的函数适配器(绑定器),
  • 它用一个可调用对象及其参数,生成一个新的可调用对象,以适应模板。
  • 接收一个函数名作为参数,生成一个新的函数。

使用std::bind可以将可调用对象和参数一起绑定,绑定后的结果使用std::function进行保存,并延迟调用到任何我们需要的时候。

2、头文件

#include <functional>

3、构造函数

template< class Fx, class... Args >
  	function<> bind (Fx&& fx, Args&...args);
  • Fx:需要绑定的可调用对象(可以是前两节课介绍的那六种,也可以是function对象)。
  • args:绑定参数列表,可以是左值、右值和参数占位符std::placeholders::_n,如果参数不是占位符,缺省为值传递,std:: ref(参数)则为引用传递。
  • std::bind()返回std::function的对象。

std::bind()的本质是仿函数。

4、demo:简单场景

  • bind场景:普通函数绑定、参数位置不同、少一个参数、多一个参数;
  • _1,_n在命名空间std::placeholders内
#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "第" << bh << "个," << message << endl;
}

int main()
{
	function<void(int, const string&)> fn1 = show;
	function<void(int, const string&)> fn2 = bind(show, placeholders::_1, placeholders::_2);
	fn1(1, "hello 西安。");
	fn2(2, "hello 西安。");

	// 调用参数位置
	function<void(const string&, int)> fn3 = bind(show, placeholders::_2, placeholders::_1);
	fn3("hello 西安。", 3);

	// 少一个参数
	function<void(const string&)> fn4 = bind(show, 4, placeholders::_1);
	fn4("hello 西安。");

	// 多一个参数
	function<void(int, const string&, int)> fn5 = bind(show, placeholders::_1, placeholders::_2);
	fn5(5, "hello 西安。", 88);
}

输出

第1个,hello 西安。
第2个,hello 西安。
第3个,hello 西安。
第4个,hello 西安。
第5个,hello 西安。

5、demo:值传递、引用传递

  • 值传递
#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "第" << bh << "个," << message << endl;
}

int main()
{

	// 少一个参数
	int n = 1;
	function<void(const string&)> fn1 = bind(show, n, placeholders::_1);
	fn1("hello 西安。");
	n = 2;
	fn1("hello 西安。");
}

输出

第1个,hello 西安。
第1个,hello 西安。

  • 引用传递
#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "第" << bh << "个," << message << endl;
}

int main()
{
	// 少一个参数
	int n = 1;
	function<void(const string&)> fn1 = bind(show, std::ref(n), placeholders::_1);
	fn1("hello 西安。");
	n = 2;
	fn1("hello 西安。");
}

输出

第1个,hello 西安。
第2个,hello 西安。

6、demo:绑定6种可调用对象

  • bind 可以把六种可调用对象的调用方式统一起来
#include <iostream>
#include <functional>
using namespace std;

// 普通函数
void show(int bh, const string& message) {
	cout << "第" << bh << "个," << message << endl;
}

struct AA	// 类中有静态成员函数。
{
	static void show(int bh, const string& message) {
		cout << "第" << bh << "个," << message << endl;
	}
};

struct BB	// 仿函数。
{
	void operator()(int bh, const string& message) {
		cout << "第" << bh << "个," << message << endl;
	}
};

struct CC	// 类中有普通成员函数。
{
	void show(int bh, const string& message) {
		cout << "第" << bh << "个," << message << endl;
	}
};

struct DD		// 可以被转换为普通函数指针的类。
{
	using Fun = void (*)(int, const string&);    // 函数指针的别名。
	operator Fun() {
		return show;	// 返回普通函数show的地址。
	}
};

int main()
{
	// 普通函数。
	function<void(int, const string&)> fn1 = bind(show, placeholders::_1, placeholders::_2);    // 绑定普通全局函数show。
	fn1(1, "长安归故里。");										                                // 用function对象调用普通全局函数show。

	// 类的静态成员函数。
	function<void(int, const string&)> fn3 = bind(AA::show, placeholders::_1, placeholders::_2);	// 绑定类的静态成员函数。
	fn3(2, "长安归故里。");										                           		   // 用function对象调用类的静态成员函数。

	// 仿函数。
	function<void(int, const string&)> fn4 = bind(BB(), placeholders::_1, placeholders::_2);		// 绑定仿函数。
	fn4(3, "长安归故里。");									                                	     // 用function对象调用仿函数。

	// 创建lambda对象。
	auto lb = [](int bh, const string& message) {
		cout << "第" << bh << "个," << message << endl;
	};
	function<void(int, const string&)> fn5 = bind(lb, placeholders::_1, placeholders::_2);			// 绑定lamba函数。
	fn5(4, "长安归故里。");										                                     // 用function对象调用lamba函数。

	// 类的非静态成员函数。
	CC cc;
	//function<void(CC&, int, const string&)> fn11 = bind(&CC::show, placeholders::_1, placeholders::_2, placeholders::_3);		// 绑定成员函数。
	//fn11(cc, 5, "我是一只傻傻鸟。");										             	                // 用function对象调用成员函数。
	function<void(int, const string&)> fn11 = bind(&CC::show, &cc, placeholders::_1, placeholders::_2);		// 绑定成员函数。
	fn11(5, "长安归故里。");											                                    // 用function对象调用成员函数。

	// 可以被转换为函数指针的类对象。
	DD dd;
	function<void(int, const string&)> fn12 = bind(dd, placeholders::_1, placeholders::_2);			// 绑定可以被转换为函数指针的类。
	fn12(6, "长安归故里。");										                                // 用function对象调用它。
}

输出

第1个,长安归故里。
第2个,长安归故里。
第3个,长安归故里。
第4个,长安归故里。
第5个,长安归故里。
第6个,长安归故里。

三、函数适配器

我们在【STL十二】适配器——容器适配器介绍过:stl适配器分为 、容器适配器、迭代器适配器、函数适配器(function adapters)。

1、函数适配器(function adapters)

  • 函数适配器:通过转换或者修改其他函数对象使其功能得到扩展。可以将预先定义的仿函数和其他数值结合在一起。
  • 函数适配器种类
    • 否定器
    • 绑定器
    • 函数指针适配器

2、很多函数适配器在c++17就被废弃了

从c++标准相关网站我们可以看到,比如,1、C++ STL 容器库 中文文档,很多函数适配器在c++17开始,就被废弃了。所以我们就不再介绍了。
在这里插入图片描述

3、bind就是一种函数适配器(绑定器)

在这里插入图片描述

参考
1、C++ STL 容器库 中文文档
2、STL教程:C++ STL快速入门
3、https://www.apiref.com/cpp-zh/cpp/header.html
4、https://en.cppreference.com/w/cpp/container
5、WIKI教程_C ++标准库_C++ Library - <iterator>
6、哔哩哔哩_系统化学习C++_C++11神器之可调用对象包装器和绑定器

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/484769.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Nessus漏洞扫描以及OpenSSH漏洞修复验证

主机IP地址资源kali192.168.200.1285GB内存/4CPUCentOS7.5192.168.200.1292GB内存/2CPU https://www.tenable.com/downloads/nessus?loginAttemptedtrue curl --request GET \--url https://www.tenable.com/downloads/api/v2/pages/nessus/files/Nessus-10.5.1-ubuntu1404_am…

云原生Istio案例实战

目录 1 Istio监控功能1.1 prometheus和grafana1.2 访问prometheus1.3 访问grafana 2 项目案例&#xff1a;bookinfo2.1 理解什么是bookinfo2.2 sidecar自动注入到微服务2.3 启动bookinfo2.4 通过ingress方式访问2.5 通过istio的ingressgateway访问2.5.1 确定 Ingress 的 IP 和端…

计算机视觉--图像拼接

图像拼接 单应性变换仿射变换图像扭曲实现图像嵌入&#xff08;图中图&#xff09; RANSAC算法算法介绍图片收集无RANSAC优化和有RANSAC优化的代码实现差别 总结 单应性变换 单应性变换是指一个平面上的点通过一个矩阵变换映射到另一个平面上的点&#xff0c;这个变换矩阵是一…

java聊天室的设计与实现代码

聊天室是一个简单的通信应用&#xff0c;可以帮助您与客户和朋友保持联系&#xff0c;并且可以让您更轻松地与其他员工联系。然而&#xff0c;您将不得不确保每个人都知道他们正在做什么。 一旦聊天室开始&#xff0c;它就会变得非常复杂&#xff0c;因为有许多用户可能会同时登…

【三十天精通Vue 3】第二十五天 Vue3 与 Axios 后端数据交互

✅创作者&#xff1a;陈书予 &#x1f389;个人主页&#xff1a;陈书予的个人主页 &#x1f341;陈书予的个人社区&#xff0c;欢迎你的加入: 陈书予的社区 &#x1f31f;专栏地址: 三十天精通 Vue 3 文章目录 引言一、Vue3 与 Axios 概述二、Axios 安装与基本使用2.1 安装 Ax…

NECCS|全国大学生英语竞赛C类|词汇和语法|语法题|时态|22:30~11:44

15题 10min 10:20&#xff5e;10:25 test2 10:25&#xff5e;10:47 test1订正 10:44&#xff5e;11:47 理论学习 涉及的语法点主要包括&#xff1a; 动词的时态和语态 非谓语动词 虚拟语气 主谓一致 倒装句 强调句 比较级 名词性从句 定语…

【SQL篇】面试篇之子查询

1303 求团队人数 # 写法1 # Write your MySQL query statement below select employee_id, count(*) over(partition by team_id) as team_size from Employee# 写法2 # Write your MySQL query statement below select employee_id, team_size from Employee e join (select t…

优雅编程,从空格、空行、缩进、注释开始

很多初学者的代码其实都不够“漂亮”&#xff0c;那是因为没有养成好的编码习惯。本篇博客以C语言为例&#xff0c;总结一些好习惯。其实&#xff0c;很多习惯都是肌肉记忆&#xff0c;举个例子&#xff1a;请你写一个程序&#xff0c;输入2个整数并输出它们的和。有些朋友可能…

springboot+vue前后端分离项目打包成jar包及运行

将 Spring Boot 和 Vue.js 项目打包成 jar 包需要按照以下步骤操作&#xff1a; 在项目的根目录中&#xff0c;使用命令行进入 Vue.js 项目的根目录&#xff0c;然后运行以下命令&#xff1a; npm run build这个命令将会构建 Vue.js 项目&#xff0c;并在项目的 dist 目录中生…

Rust-Rocket框架笔记

Rust-Rocket框架笔记 Rocket-Learn-docRocket Addr视频地址 What is RocketQuickStart下载Rocket-Rust运行Rust-Rocket-Hello-错误-端口占用解决查看端口占用情况添加Rocket.toml配置文件更改Rocket默认启动端口启动成功 GetStart-Hello world创建项目cargoIDEA 添加依赖添加Ro…

使用eclipse创建一个图书管理系统(2)---------逻辑的实现

就像使用C语言写代码一样。比如要用​​​​​​C语言写一个小游戏的代码&#xff0c;我们的逻辑实现是在哪里实现的啊&#xff1f;是不是在一个test.c源文件里面啊&#xff1f;没错&#xff0c;就是的&#xff01;在之前的文章里&#xff0c;我说过我定义的Main函数就像C语言里…

【《中国工业经济》数据复现】数字化转型与企业分工:专业化还是纵向一体化

一.研究内容 本文使用机器学习方法刻画微观企业数字化水平&#xff0c;并在构建数理模型的基础上实证考察了企业数字化转型对企业分工的影响及其机理。结果表明&#xff0c;企业数字化转型显著提升了中国上市企业专业化分工水平。机制分析表明&#xff0c;数字化转型对企业专业…

实现chatgpt自然对话

1.概述 ChatGPT是当前自然语言处理领域的重要进展之一&#xff0c;通过预训练和微调的方式&#xff0c;ChatGPT可以生成高质量的文本&#xff0c;可应用于多种场景&#xff0c;如智能客服、聊天机器人、语音助手等。本文将详细介绍ChatGPT的原理、实战演练和流程图&#xff0c…

C/C++每日一练(20230503)

目录 1. 输出最长的递增数字字符串 &#x1f31f;&#x1f31f; 2. 缺失的第一个正数 &#x1f31f;&#x1f31f;&#x1f31f; 3. 最大矩形 &#x1f31f;&#x1f31f;&#x1f31f; &#x1f31f; 每日一练刷题专栏 &#x1f31f; Golang每日一练 专栏 Python每日…

SaaS模医学检验信息管理系统源码,系统预设几十种报告模板,可在几分钟内批量生成报告

实验室信息管理系统云LIS源码 SaaS模式运维管理系统 云LIS系统源码是一款全面的实验室信息管理系统源码&#xff0c;其主要功能包括样本管理、检测项目管理、质控管理、报告管理、数据分析、两癌筛查等多个方面。具有独立的配套SaaS模式运维管理系统&#xff0c;支持远程运维&…

Redis高频面试题,使用场景

一、缓存 1、什么是缓存穿透 ? 怎么解决 ? 缓存穿透 查询一个不存在的数据&#xff0c;mysql查询不到数据也不会直接写入缓存&#xff0c;就会导致每次请求都查数据库。 解决 方案一&#xff1a;缓存空数据&#xff0c;查询返回的数据为空&#xff0c;仍把这个空结果进行…

《花雕学AI》28:革命性的 ChatGPT for SEO——让您的排名飙升 50%!

引言&#xff1a; 如果您想写篇有吸引力的文章&#xff0c;或者您是一个博客和网站的拥有者&#xff0c;那么您一定知道 SEO&#xff08;搜索引擎优化&#xff09;的重要性。SEO 可以帮助您提高相应的流量、转化率和收入&#xff0c;但是 SEO 也是一个复杂和耗时的过程&#x…

【开源项目】Dynamic-Tp核心流程源码解读

序.介绍 dynamic-tp 是一款动态线程池组件&#xff0c;可以实现线程池的实时动态调参及监控报警&#xff0c;线程池配置放在配置中心统一管理&#xff0c;达成业务代码零侵入&#xff0c;支持多配置中心的选择和常见的第三方组件的线程池的集成管理。 官网: https://dynamict…

C++学习day--01 C生万物

1、C/C学习中遇到的问题&#xff1a; 1. 大部分初学者&#xff0c;学习 C/C 都是从入门到放弃。 C/C太难吗&#xff1f; 2. 90% 以上的初学者&#xff0c;学完 C/C 以后&#xff0c;考试完了&#xff0c;书看完了&#xff0c; 但还是不会做项目 是学的不够好吗&#xff1…

基于KZG多项式承诺方案的RLN

1. 引言 RLN——Rate-Limiting Nullifier为PSE团队主导的项目&#xff0c;源自&#xff1a; Barry White Hat 2019年博客 Semaphore RLN, rate limiting nullifier for spam prevention in anonymous p2p setting RLN&#xff08;Rate-Limiting Nullifier&#xff09;是一种…