C++ STL:函数对象

简介: C++ STL:函数对象

Part 5:函数对象

函数对象(仿函数):重载了函数调用运算符的类创建的对象,将所有的函数形式进行统筹,实现容器的定制化操作。

1、function

函数类型的容器。包装具有指定函数调用签名的任意可复制构造类型的可调用对象

  • 普通函数 | 成员函数
  • 函数指针
  • 重载了函数调用运算符的类创建的对象
// std::function 相当于是函数类型的容器
 // 1、std::bind + std::function 绑定普通函数
 std::function<> f = std::bind(add, 1, 2);
 cout << f() << endl;        // 输出 3
 std::function<int(int)> f1 = std::bind(add, 1, _1);
 cout << f1(10) << endl;     // 输出 11
 std::function<int(int, int)> f2 = std::bind(add, 1, _2);  // 不推荐,这里只为了演示
 cout << f2(10, 11) << endl;  // 输出 12
 ///2、std::bind + std::function 绑定普通函数
 std::function<> f = std::bind(&MyTest::add, &mytest, 1, 2);
 cout << f() << endl;         // 输出 3
 std::function<int(int)> f1 = std::bind(&MyTest::add, &mytest, _1, 2);
 cout << f1(10) << endl;      // 输出 12
 std::function<int(int, int)> f2 = std::bind(&MyTest::add, mytest, _1, _1);
 cout << f2(10, 20) << endl;  // 输出 20

2、function + bind 多态

关于 bind 的方法:见我的博客 C++ STL:适配器 Adapter 中的函数适配器部分。

多态的实现机制

  • 继承 + 虚函数
  • 回调函数:组件 + 接口

function + bind 通过函数对象,注册回调函数,实现多态,取代继承 + 虚函数实现多态的机制。相比于虚函数多态机制,回调函数多态机制具有灵活性:

  • 不再需要虚函数。虚函数本身就很严格
  • 不再需要继承。聚合
  • 非常直接,想调用哪个方法,注册对应的回调函数即可

例如:计算图形的面积

测试1:继承 + 虚函数实现的多态

#include <math.h>
 #include <iostream>
 #include <string>
 #include <memory>
 using std::cout;
 using std::endl;
 using std::string;
 using std::unique_ptr;
 class Figure {
 public:
     virtual void display() const =0;
     virtual double area()=0;
     virtual ~Figure() {}
 };
 class Circle: public Figure {
 public:
     Circle(double radius)
     : _radius(radius)
     {}
     void display() const
     {   cout << "circle";   }
     double area() {
         return 3.14159 * _radius * _radius;
     }
     ~Circle() { cout << "~Circle()" << endl;    }
 private:
     double _radius;
 };
 class Rectangle: public Figure {
 public:
     Rectangle(double length, double width)
     : _length(length)
     , _width(width)
     {}
     ~Rectangle() { cout << "~Rectangle()" << endl; }
     void display() const
     {   cout << "rectangle";    }
     double area()
     {   return _length * _width;    }
 private:
     double _length;
     double _width;
 };
 class Triangle: public Figure {
 public:
     Triangle(double a, double b, double c)
     : _a(a), _b(b), _c(c) {}
     ~Triangle() {   cout << "~Triangle()" << endl;  }
     void display() const
     {   cout << "triangle";}
     double area() {
         double p = (_a + _b + _c) / 2;
         return sqrt(p * (p - _a) * (p - _b) * (p - _c));
     }
 private:
     double _a;
     double _b;
     double _c;
 };
 void display(Figure * fig) {
     fig->display();
     cout << "的面积是:" << fig->area() << endl;
 }
 int main(void) {
     Rectangle rectangle(10, 20);
     Circle circle(10);
     Triangle triangle(3, 4, 5);
     display(&rectangle);
     display(&circle);
     display(&triangle);
     return 0;
 }

测试2:function + bind 实现的多态

#include <math.h>
 #include <iostream>
 #include <string>
 #include <memory>
 #include <functional>
 using std::cout;
 using std::endl;
 using std::string;
 using std::unique_ptr;
 /* 
     通过 std::function + std::bind 取代继承 + 虚函数, 实现多态的机制
     这种多态机制的灵活性体现在:
     1. 不再需要虚函数,虚函数本身就是很严格
     2. 不再需要继承
     3. 非常直接,想调用哪个方法,注册对应的回调函数即可
 */
 // Figure 不是抽象类,而是一个具体类
 class Figure {
 public:
     using DisplayCallback = std::function<void()>;
     using AreaCallback = std::function<double()>;
     Figure() {  cout << "Figure()" << endl;};
     ~Figure() {}
     void setCallbacks(DisplayCallback && cb1, AreaCallback && cb2) {
         _displayCallback = std::move(cb1);
         _areaCallback = std::move(cb2);
     }
     void handleDisplayCallback() {
         if(_displayCallback){
             _displayCallback();
         }
     }
     double handleAreaCallback() {
         if(_areaCallback) 
             return  _areaCallback();
         else 
             return 0;
     }
     DisplayCallback _displayCallback;
     AreaCallback _areaCallback;
 };
 // 不再使用继承
 class Circle {
 public:
     Circle(double radius)
     : _radius(radius)
     {}
     void display() const
     {   cout << "circle";   }
     double area() {
         return 3.14159 * _radius * _radius;
     }
     ~Circle() { cout << "~Circle()" << endl;    }
 private:
     double _radius;
 };
 class Rectangle {
 public:
     Rectangle(double length, double width)
     : _length(length), _width(width) {}
     ~Rectangle() { cout << "~Rectangle()" << endl; }
     void print() const
     {   cout << "rectangle";    }
     double calcArea()
     {   return _length * _width;    }
 private:
     double _length;
     double _width;
 };
 class Triangle {
 public:
     Triangle(double a, double b, double c)
     : _a(a), _b(b), _c(c) {}
     ~Triangle() {   cout << "~Triangle()" << endl;  }
     void show() const
     {   cout << "triangle";}
     double getArea() {
         double p = (_a + _b + _c) / 2;
         return sqrt(p * (p - _a) * (p - _b) * (p - _c));
     }
 private:
     double _a;
     double _b;
     double _c;
 };
 void display(Figure & fig) {
     fig.handleDisplayCallback();
     cout << "的面积是:" << fig.handleAreaCallback() << endl;
 }
 int main(void) {
     Rectangle rectangle(10, 20);
     Circle circle(10);
     Triangle triangle(3, 4, 5);
     // 1、注册回调函数
     // 2、执行回调函数
     Figure fig;
     fig.setCallbacks(
         std::bind(&Rectangle::print, &rectangle),
         std::bind(&Rectangle::calcArea, &rectangle));
     display(fig);
     cout << endl;
     fig.setCallbacks(
         std::bind(&Circle::display, &circle),
         std::bind(&Circle::area, &circle));
     display(fig);
     cout << endl;
     fig.setCallbacks(
         std::bind(&Triangle::show, &triangle),
         std::bind(&Triangle::getArea, &triangle));
     display(fig);
     return 0;
 }
相关文章
|
11天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
17 1
|
24天前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
42 7
|
21小时前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
32 18
|
21小时前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
28 13
|
2月前
|
存储 编译器 C语言
【c++丨STL】vector的使用
本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
73 4
|
2月前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
86 5
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
67 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
120 5
|
2月前
|
存储 算法 Linux
【c++】STL简介
本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
66 0
|
27天前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
33 0