服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C++11学习之包装器解析

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

2023-03-02 15:30张小姐的猫 C/C++

function包装器 也叫作适配器。C++中的function本质是一个类模板,也是一个包装器。本文就来和大家聊聊我们为什么需要function呢

概念

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

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

包装器定义式:

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

模板参数说明:

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
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函数。

代码如下:

?
1
2
3
4
5
6
7
8
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);
}

在不使用包装器,直接传入对象的时候,会实例化出三份

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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;
}

输出结果如下:

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

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

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

包装后代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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;
}

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

例题:求解逆波兰表达式

题目:

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

解题思路:

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

此处的包装器:

  • 建立各个运算符与其对应需要执行的函数之间的映射关系,当需要执行某一运算时就可以直接通过运算符找到对应的函数进行执行
  • 当运算类型增加时,就只需要建立新增运算符与其对应函数之间的映射关系即可(其他代码不用动)
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
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函数模板的原型如下:

?
1
2
3
4
template <class Fn, class... Args>
/* unspecified */ bind(Fn&& fn, Args&&... args);
template <class Ret, class Fn, class... Args>
/* unspecified */ bind(Fn&& fn, Args&&... args);

模板参数说明:

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

调用bind的一般形式

?
1
auto newCallable = bind(callable, arg_list);

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

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

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

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

举例:

?
1
2
3
4
5
using namespace placeholders;
int x = 2, y = 10;
Div(x, y);
 
auto bindFun1 = bind(Div, _1, _2);

bind 绑定固定参数

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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。比如:

?
1
2
3
4
5
6
7
8
9
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 对象:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
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 的位置交换一下就行了:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
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包装器的资料请关注服务器之家其它相关文章!

原文链接:https://blog.csdn.net/qq_42996461/article/details/128888581

延伸 · 阅读

精彩推荐
  • C/C++c++中const的使用详解

    c++中const的使用详解

    本篇文章是对c++中的const的应用进行了详细的分析介绍,需要的朋友参考下 ...

    C++教程网2152020-11-27
  • C/C++C++inline函数的特性你了解吗

    C++inline函数的特性你了解吗

    这篇文章主要为大家详细介绍了C++的inline函数,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来...

    guo541111072022-10-24
  • C/C++C++遍历文件夹获取文件列表

    C++遍历文件夹获取文件列表

    这篇文章主要为大家详细介绍了C++遍历文件夹获取文件列表的相关资料,感兴趣的小伙伴们可以参考一下...

    leno米雷4182021-04-02
  • C/C++C++链式二叉树深入分析

    C++链式二叉树深入分析

    二叉树的链式存储结构是指,用链表来表示一棵二叉树,即用链来指示元素的逻辑关系。通常的方法是链表中每个结点由三个域组成,数据域和左右指针域...

    沙漠下的胡杨10292022-12-20
  • C/C++深入了解C语言栈的创建

    深入了解C语言栈的创建

    栈只允许在一端进行插入或删除操作的线性表。首先栈是一种线性表,但是限定这种线性表只能在某一端进行插入和删除操作,这篇文章主要介绍了C语言对...

    小尹同学⁣9812021-12-08
  • C/C++C语言函数栈帧的创建和销毁介绍

    C语言函数栈帧的创建和销毁介绍

    大家好,本篇文章主要讲的是C语言函数栈帧的创建和销毁介绍,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下...

    诚挚的乔治7722022-07-16
  • C/C++C语言中的pause()函数和alarm()函数以及sleep()函数

    C语言中的pause()函数和alarm()函数以及sleep()函数

    这篇文章主要介绍了C语言中的pause()函数和alarm()函数以及sleep()函数,是C语言入门学习中的基础知识,需要的朋友可以参考下...

    C语言教程网6652021-03-11
  • C/C++C语言解决青蛙跳台阶问题(升级版)

    C语言解决青蛙跳台阶问题(升级版)

    所谓的青蛙跳台阶问题,就是指一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法。本文将用C语言解决这一问...

    飞向星的客机4842022-09-06