返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++11学习之包装器解析
  • 724
分享到

C++11学习之包装器解析

C++11包装器解析C++11包装器C++ 包装器 2023-02-13 12:02:35 724人浏览 薄情痞子
摘要

目录概念类型统一例题:求解逆波兰表达式包装器的意义bind 包装器bind 绑定固定参数调整传参顺序bind包装器的意义概念 function包装器 也叫作适配器。c++中的func

概念

function包装器 也叫作适配器。c++中的function本质是一个类模板,也是一个包装器。

那么我们来看看,我们为什么需要function呢?

包装器定义式:

// 类模板原型如下
template <class T> function;     // undefined
template <class Ret, class... Args>
class function<Ret(Args...)>;

模板参数说明:

  • Ret: 是被包装的可调用对象的返回值类型
  • Args... :是被包装的可调用对象的形参类型

function包装器可以对可调用对象进行包装,包括函数指针、函数名、仿函数(函数对象)、lambda表达式

int f(int a, int b)
{
	return a + b;
}

struct Functor
{
public:
	int operator() (int a, int b)
	{
		return a + b;
	}
};

class Plus
{
public:
	//静态 vs 非静态
	static int plusi(int a, int b)
	{
		return a + b;
	}

	double plusd(double a, double b)
	{
		return a + b;
	}
};

int main()
{
	function<int(int, int)> f1 = f;
	f1(1, 2);

	function<int(int, int)> f2 = Functor();
	f2(1, 2);

	function<int(int, int)> f3 = Plus::plusi;
	f3(1, 2);

	//非静态成员函数    要 + 对象
	function<double(Plus, double, double)> f4 = &Plus::plusd;
	f4(Plus(), 1.1, 2.2);

	return 0;
}

注意事项:

取静态成员函数的地址可以不用取地址运算符“&”,但取非静态成员函数的地址必须使用取地址运算符“&”

非静态成员函数在调用的时候要 + 对象,因为非静态成员函数的第一个参数是隐藏this指针,所以在包装时需要指明第一个形参的类型为类的类型。

类型统一

对于以下函数模板useF:

  • 传入该函数模板的第一个参数可以是任意的可调用对象,比如函数指针、仿函数、lambda表达式等
  • useF中定义了静态变量count,并在每次调用时将count的值和地址进行了打印,可判断多次调用时调用的是否是同一个useF函数。

代码如下:

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;

	//lambda表达式
	cout << useF([](double d)->double{return d / 4; }, 11.11) << endl;
	return 0;
}

输出结果如下:

由于函数指针、仿函数、lambda表达式是不同的类型,因此useF函数会被实例化出三份,三次调用useF函数所打印count的地址也是不同的。

  • 但实际这里其实没有必要实例化出三份useF函数,因为三次调用useF函数时传入的可调用对象虽然是不同类型的,但这三个可调用对象的返回值和形参类型都是相同的
  • 这时就可以用包装器分别对着三个可调用对象进行包装,然后再用这三个包装后的可调用对象来调用useF函数,这时就只会实例化出一份useF函数
  • 根本原因就是因为包装后,这三个可调用对象都是相同的function类型,因此最终只会实例化出一份useF函数,该函数的第一个模板参数的类型就是function类型的

包装后代码如下:

int main()
{
	// 函数指针
	function<double(double)> f1 = f;
	cout << useF(f1, 11.11) << endl;

	// 函数对象
	function<double(double)> f2 = Functor();
	cout << useF(f2, 11.11) << endl;

	// lamber表达式
	function<double(double)> f3 = [](double d)->double { return d / 4; };
	cout << useF(f3, 11.11) << endl;

	return 0;
}

例题:求解逆波兰表达式

题目:

解题思路:

  • 首先定义一个栈,依次遍历所给字符串
  • 遇到数字,直接入栈,遇到操作符,则从栈定抛出两个数字进行对应的运算,并将运算后得到的结果压入栈中
  • 所给字符串遍历完毕后,栈顶的数字就是逆波兰表达式的计算结果

此处的包装器:

  • 建立各个运算符与其对应需要执行的函数之间的映射关系,当需要执行某一运算时就可以直接通过运算符找到对应的函数进行执行
  • 当运算类型增加时,就只需要建立新增运算符与其对应函数之间的映射关系即可(其他代码不用动)
class Solution {
public:
    int evalRPN(vector<string>& tokens) {
        stack<long long> st;
        map<string, function<long long(long long, long long)>> opfuncMap = 
        {
            //自动构造pair ~ 初始化列表构造
            {"+", [](long long a , long long b){return a + b;}},
            {"-", [](long long a , long long b){return a - b;}},
            {"*", [](long long a , long long b){return a * b;}},
            {"/", [](long long a , long long b){return a / b;}},
        };


        for(auto& str : tokens)
        {
            if(opfuncMap.count(str))
            {
                //操作符 :出栈(先出右,再出左)
                long long right = st.top();
                st.pop();
                long long left = st.top();
                st.pop();
                st.push(opfuncMap[str](left, right));
            }
            else
            {
                //操作数:入栈
                st.push(stoll(str));
            }
        }
        return st.top();
    }
};

包装器的意义

将可调用对象的类型进行统一,便于我们对其进行统一化管理。

包装后明确了可调用对象的返回值和形参类型,更加方便使用者使用。

bind 包装器

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

bind函数模板的原型如下:

template <class Fn, class... Args>
 bind(Fn&& fn, Args&&... args);
template <class Ret, class Fn, class... Args>
 bind(Fn&& fn, Args&&... args);

模板参数说明:

  • fn:可调用对象
  • args...:要绑定的参数列表:值或占位符

调用bind的一般形式

auto newCallable = bind(callable, arg_list);

callable:需要包装的可调用对象

newCallable:生成的新的可调用对象

arg_list:逗号分隔的参数列表,对应给定的 callable 的参数,当调用 newCallable时,newCallable 会调用 callable,并传给它 arg_list 中的参数

_1 _2 ... 是定义在placeholders命名空间中,代表绑定函数对象的形参;_1代表第一个形参,_2代表第二个形参 …

举例:

using namespace placeholders;
int x = 2, y = 10;
Div(x, y);

auto bindFun1 = bind(Div, _1, _2);

bind 绑定固定参数

原本传入的参数要求是要3个,现在只需要输入两个参数,因为绑定了固定的函数对象

using namespace placeholders;

class Sub
{
public:
	int sub(int a, int b)
	{
		return a - b;
	}
};

int main()
{
	//function<int(Sub, int, int)> fsub = &Sub::sub;
	function<int(int, int)> fsub = bind(&Sub::sub, Sub(), _1, _2);
}

想把Mul函数的第三个参数固定绑定为1.5,可以在绑定时将参数列表的placeholders::_3设置为1.5。比如:

int Mul(int a, int b, int rate)
{
	return a * b * rate;
}

int main()
{
	function<int(int, int)> fmul = bind(Mul, _1, _2, 1.5);
}

调整传参顺序

对于 Sub 类中的 sub 函数,因为 sub 的第一个参数是隐藏的 this 指针,如果想要在调用 sub 时不用对象进行调用,那么可以将 sub 的第一个参数固定绑定为一个 Sub 对象:

using namespace placeholders;

class Sub
{
public:
	int sub(int a, int b)
	{
		return a - b;
	}
};
int main()
{
	//绑定固定参数
	function<int(int, int)> func = bind(&Sub::sub, Sub(), _1, _2);
	cout << func(1, 2) << endl; 
	return 0;
}

此时调用对象时就只需要传入用于相减的两个参数,因为在调用时会固定帮我们传入一个匿名对象给 this 指针。

如果想要将 sub 的两个参数顺序交换,那么直接在绑定时将 _1 和_2 的位置交换一下就行了:

using namespace placeholders;

class Sub
{
public:
    int sub(int a, int b)
    {
        return a - b;
    }
};
int main()
{
    //绑定固定参数
    function<int(int, int)> func = bind(&Sub::sub, Sub(), _2, _1);
    cout << func(1, 2) << endl; 
    return 0;
}

其原理:第一个参数会传给_1,第二个参数会传给 _2,因此可以在绑定时通过控制 _n 的位置,来控制第 n 个参数的传递位置

bind包装器的意义

1.将一个函数的某些参数绑定为固定的值,让我们在调用时可以不用传递某些参数。

2.可以对函数参数的顺序进行灵活调整。

以上就是C++11学习之包装器解析的详细内容,更多关于C++11包装器的资料请关注编程网其它相关文章!

--结束END--

本文标题: C++11学习之包装器解析

本文链接: https://lsjlt.com/news/195811.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C++11学习之包装器解析
    目录概念类型统一例题:求解逆波兰表达式包装器的意义bind 包装器bind 绑定固定参数调整传参顺序bind包装器的意义概念 function包装器 也叫作适配器。C++中的func...
    99+
    2023-02-13
    C++11包装器解析 C++11包装器 C++ 包装器
  • 【C++】C++11 ~ 包装器解析
    🌈欢迎来到C++专栏~~包装器解析 ...
    99+
    2023-09-12
    c++ java 开发语言
  • C++11系列学习之可调用对象包装器和绑定器
    目录旧标准的缺陷繁杂的调用对象问题浮出水面std::function小试牛刀std::bind作用占位符高级用法配合使用旧标准的缺陷 学习新标准的语法之前,先来聊聊旧标准存在的缺陷,...
    99+
    2024-04-02
  • C++11学习之多线程的支持详解
    目录C++11中的多线程的支持1.C++11中的原子类型1.1 原子类型的接口1.2简单自旋锁的实现2.提高并行程度2.1 memory_order的参数2.2 release-ac...
    99+
    2023-02-06
    C++11多线程支持 C++11多线程
  • C++11系列学习之类型推导
    目录auto类型推导auto基本用法auto 推导规则auto 的限制auto 适用场景decltype 类型推导decltype 基本用法decltype 推导规则decltype...
    99+
    2024-04-02
  • ​​C++11系列学习之Lambda表达式
    目录一、为什么要有lambda表达式?二、使用语法捕获列表mutable影响lambda表达式std::bind和lambda表达式结合三、std::function 和lambda...
    99+
    2024-04-02
  • C++11新增的包装器详解
    目录functionbindfunction 目前,我们的知识深度已知的可调用对象类型有: 函数指针仿函数 / 函数对象lambda表达式 现在我们有一个函数模板 templa...
    99+
    2024-04-02
  • C++11系列学习之列表初始化
    目录前言:旧标准初始化方式C++11标准初始化方式初始化列表技术细节总结前言: 由于旧标准初始化方式太过繁杂,限制偏多,因此在新标准中统一了初始化方式,为了让初始化具有确定的效果,于...
    99+
    2024-04-02
  • 深入解析C++11 lambda表达式/包装器/线程库
    目录零、前言一、lambda表达式1、lambda的引入2、lambda表达式语法3、捕获列表说明4、函数对象与lambda表达式二、包装器1、function包装器2、bind 概...
    99+
    2024-04-02
  • c#基础学习之封装
    作为一个初级GIS程序员,关于封装那些宏观的概念暂且不提,编程经常面对的就是“字段,属性,方法”,这也是面向对象的基本概念之一。 1.字段 通常定义为private,表示类的状态信息...
    99+
    2022-11-15
    c# 封装
  • C++11学习之右值引用和移动语义详解
    目录左值引用与右值引用1、左值与右值2、纯右值、将亡值3、左值引用与右值引用4、右值引用和 std::move 使用场景引用限定符const 和引用限定符移动语义—std...
    99+
    2023-02-23
    C++11右值引用 移动语义 C++11右值引用 C++11 移动语义
  • python3学习之装饰器
    #定义装饰器,outer参数是函数,返回也是函数 #作用:在函数执行前和执行后分别附加额外功能 def  outer(func):     def  inner(*args, **kwargs):         print("log") ...
    99+
    2023-01-31
  • C++学习之线程详解
    目录开篇线程的状态多线程的构建计算时间一、程序运行时间二、chrono共享资源和互斥锁condition_variable线程池总结开篇 多线程是开发中必不可少的,往往我们需要多个任...
    99+
    2024-04-02
  • Python学习之包与模块详解
    目录什么是 Python 的包与模块包的身份证如何创建包创建包的小练习包的导入 - import模块的导入 - from…import导入子包及子包函数的调用导入主包及...
    99+
    2024-04-02
  • C++学习之cstdbool和cstddef头文件封装源码分析
    目录引言stdbool.hcstdbool实现C语言的原生实现stdbool.h小结stddef.h常量NULL的定义类型的定义offsetof宏引言 cstdbool是C++对st...
    99+
    2024-04-02
  • nodejs模块学习之connect解析
    nodejs 发展很快,从 npm 上面的包托管数量就可以看出来。不过从另一方面来看,也是反映了 nodejs 的基础不稳固,需要开发者创造大量的轮子来解决现实的问题。 知其然,并知其所以然这是程序员的天性...
    99+
    2022-06-04
    模块 nodejs connect
  • Python学习之装饰器与类的装饰器详解
    目录装饰器装饰器的定义装饰器的用法类中的装饰器类的装饰器 - classmethod类的装饰器 - staticmethod类的装饰器 - property通过学习装饰器可以让我们更...
    99+
    2024-04-02
  • C++11 学习笔记之std::function和bind绑定器
    std::function       C++中的可调用对象虽然具有比较统一操作形式(除了类成员指针之外,都是后面加括号进行调用),...
    99+
    2024-04-02
  • C++学习之异常机制详解
    目录1. 异常处理机制介绍2. 如何抛出异常和捕获异常2.1 抛出异常2.2 捕获异常3. 如何实现自己的异常4. 注意事项5. 面试常问的题目6. 答案7. 总结1. 异常处理机制...
    99+
    2023-05-15
    C++异常机制 C++异常
  • C++学习之命名空间详解
    目录1.命名空间的定义和使用2.命名空间嵌套3.命名空间别名4.标准命名空间总结C++中,命名空间(namespace)是一个重要的概念。命名空间可以为函数、变量、类等定义作用域,以...
    99+
    2023-05-18
    C++命名空间定义 C++命名空间使用 C++命名空间
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作