享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能。这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的方式。
享元模式尝试重用现有的同类对象,如果未找到匹配的对象,则创建新对象。

我们将通过创建 5 个对象来画出 20 个分布于不同位置的圆来演示这种模式。由于只有 5 种可用的颜色,所以 color 属性被用来检查现有的 Circle 对象。

意图

运用共享技术有效地支持大量细粒度的对象。

主要解决

在有大量对象时,有可能会造成内存溢出,我们把其中共同的部分抽象出来,如果有相同的业务请求,直接返回在内存中已有的对象,避免重新创建。

何时使用

  1. 系统中有大量对象。
  2. 这些对象消耗大量内存。
  3. 这些对象的状态大部分可以外部化。
  4. 这些对象可以按照内蕴状态分为很多组,当把外蕴对象从对象中剔除出来时,每一组对象都可以用一个对象来代替。
  5. 系统不依赖于这些对象身份,这些对象是不可分辨的。

如何解决

用唯一标识码判断,如果在内存中有,则返回这个唯一标识码所标识的对象。

关键代码

用 HashMap 存储这些对象。

应用实例

  1. JAVA 中的 String,如果有则返回,如果没有则创建一个字符串保存在字符串缓存池里面。
  2. 数据库的数据池。

优点

大大减少对象的创建,降低系统的内存,使效率提高。

缺点

提高了系统的复杂度,需要分离出外部状态和内部状态,而且外部状态具有固有化的性质,不应该随着内部状态的变化而变化,否则会造成系统的混乱。

使用场景

  1. 系统有大量相似对象。
  2. 需要缓冲池的场景。

注意事项

  1. 注意划分外部状态和内部状态,否则可能会引起线程安全问题。
  2. 这些类必须有一个工厂对象加以控制。

示例

图形工厂类制作不同图形,有五种颜色的图形,每种只需要一份实例,只需要设置坐标、半径即可绘制,无需重新实例化,可以在输出结果中看到只初始化了五个实例。
image.png

  1. #include "pch.h"
  2. #include <stdlib.h>
  3. #include <iostream>
  4. #include <vector>
  5. #include <unordered_map>
  6. class Shape
  7. {
  8. public:
  9. virtual void Draw() = 0;
  10. };
  11. class Circle : public Shape
  12. {
  13. public:
  14. Circle(std::string color)
  15. {
  16. std::cout << "A circle has been created" << std::endl;
  17. this->color = color;
  18. }
  19. void SetX(int x)
  20. {
  21. this->x = x;
  22. }
  23. void SetY(int y)
  24. {
  25. this->y = y;
  26. }
  27. void SetRadius(int radius)
  28. {
  29. this->radius = radius;
  30. }
  31. virtual void Draw()
  32. {
  33. std::cout << "Circle: [Color:" << color.c_str()
  34. << ", x:" << x
  35. << ", y:" << y
  36. << ", radius:" << radius
  37. << std::endl;
  38. }
  39. private:
  40. std::string color;
  41. int x;
  42. int y;
  43. int radius;
  44. };
  45. class CircleFactory
  46. {
  47. public:
  48. static Circle GetCircle(std::string color)
  49. {
  50. std::unordered_map<std::string, Circle>::iterator it = circleMap.find(color);
  51. if (it != circleMap.end())
  52. {
  53. return it->second;
  54. }
  55. else
  56. {
  57. Circle circle(color);
  58. circleMap.insert(std::make_pair(color, circle));
  59. return circle;
  60. }
  61. return Circle("x");
  62. }
  63. private:
  64. static std::unordered_map<std::string, Circle> circleMap;
  65. };
  66. std::unordered_map<std::string, Circle> CircleFactory::circleMap;
  67. int main()
  68. {
  69. std::vector<std::string> colors = { "Red", "White", "Black", "Green", "Blue" };
  70. CircleFactory* circleFactory = new CircleFactory();
  71. for (int i = 0; i < 20; ++i)
  72. {
  73. Circle circle = circleFactory->GetCircle(colors[rand() % 5]);
  74. circle.SetX(rand() % 100);
  75. circle.SetY(rand() % 100);
  76. circle.SetRadius(rand() % 100);
  77. circle.Draw();
  78. }
  79. }
>>>
A circle has been created
Circle: [Color:White, x:67, y:34, radius:0
A circle has been created
Circle: [Color:Blue, x:24, y:78, radius:58
A circle has been created
Circle: [Color:Black, x:64, y:5, radius:45
Circle: [Color:White, x:27, y:61, radius:91
A circle has been created
Circle: [Color:Red, x:42, y:27, radius:36
Circle: [Color:White, x:4, y:2, radius:53
Circle: [Color:Black, x:82, y:21, radius:16
A circle has been created
Circle: [Color:Green, x:95, y:47, radius:26
Circle: [Color:White, x:38, y:69, radius:12
Circle: [Color:Black, x:99, y:35, radius:94
Circle: [Color:Green, x:11, y:22, radius:33
Circle: [Color:Green, x:64, y:41, radius:11
Circle: [Color:Green, x:68, y:47, radius:44
Circle: [Color:Black, x:57, y:37, radius:59
Circle: [Color:Green, x:41, y:29, radius:78
Circle: [Color:White, x:35, y:90, radius:42
Circle: [Color:Green, x:6, y:40, radius:42
Circle: [Color:Blue, x:48, y:46, radius:5
Circle: [Color:Red, x:29, y:70, radius:50
Circle: [Color:White, x:1, y:93, radius:48