原文: https://howtodoinjava.com/design-patterns/structural/flyweight-design-pattern/

根据 GoF 定义,享元设计模式可以使用对象共享来有效地支持大量细粒度对象。 权重是共享对象,可以同时在多个上下文中使用。 享元在每种情况下都充当独立对象。

1.何时使用享元设计模式

我们可以在以下情况下使用享元模式:

  • 当我们需要大量类似的对象时,这些对象只有几个参数是唯一的,而大多数东西通常都是通用的。
  • 我们需要通过创建更少的对象并将它们共享来控制大量对象的内存消耗。

2.外部和固有属性

享元物体本质上具有两种属性 - 内部属性和外部属性。

固有状态属性存储/共享在享元对象中,并且与享元的上下文无关。 作为最佳实践,我们应该使固有状态不可变

外部状态会随着重量级环境的变化而变化,这就是为什么它们无法共享的原因。 客户端对象保持外部状态,它们需要在对象创建过程中将其传递给享元对象。

享元设计模式 - 图1

享元模式

3.享元模式的真实示例

  • 假设我们有一个,可以加/不加笔芯。 笔芯可以是任何颜色,因此可以使用钢笔创建具有 N 种颜色的图形。
    这里Pen可以是具有refill作为外部属性的享元对象。 所有其他属性(例如笔身,指针等)可以是所有笔所共有的固有属性。 一支笔只能通过其笔芯颜色来区分,没有别的。
    所有需要访问红笔的应用模块 - 都可以使用红笔的同一实例(共享对象)。 仅当需要使用不同颜色的笔时,应用模块才会从享元工厂要求另一只笔。

  • 在编程中,我们可以将java.lang.String常量视为享元对象。 所有字符串都存储在字符串池中,如果我们需要带有某些内容的字符串,那么运行时将返回对该池中已存在的字符串常量的引用(如果有)。

  • 在浏览器中,我们可以在网页的多个位置使用图片。 浏览器将仅加载一次图像,而其他浏览器将重用缓存中的图像。 现在图像是相同的,但已在多个地方使用。 URL 是固有属性,因为它是固定且可共享的。 图像的位置坐标,高度和宽度是外部属性,它们根据必须渲染的位置(上下文)而变化。

4.享元设计模式示例

在给定的示例中,我们正在构建一个笔刷应用,客户可以在这三种类型上使用画笔 - THICKTHINMEDIUM。 所有粗(细或中等)画笔将以完全相似的方式绘制内容 - 仅内容颜色会有所不同。

  1. public interface Pen
  2. {
  3. public void setColor(String color);
  4. public void draw(String content);
  5. }
  1. public enum BrushSize {
  2. THIN, MEDIUM, THICK
  3. }
  1. public class ThickPen implements Pen {
  2. final BrushSize brushSize = BrushSize.THICK; //intrinsic state - shareable
  3. private String color = null; //extrinsic state - supplied by client
  4. public void setColor(String color) {
  5. this.color = color;
  6. }
  7. @Override
  8. public void draw(String content) {
  9. System.out.println("Drawing THICK content in color : " + color);
  10. }
  11. }
  1. public class ThinPen implements Pen {
  2. final BrushSize brushSize = BrushSize.THIN;
  3. private String color = null;
  4. public void setColor(String color) {
  5. this.color = color;
  6. }
  7. @Override
  8. public void draw(String content) {
  9. System.out.println("Drawing THIN content in color : " + color);
  10. }
  11. }
  1. public class MediumPen implements Pen {
  2. final BrushSize brushSize = BrushSize.MEDIUM;
  3. private String color = null;
  4. public void setColor(String color) {
  5. this.color = color;
  6. }
  7. @Override
  8. public void draw(String content) {
  9. System.out.println("Drawing MEDIUM content in color : " + color);
  10. }
  11. }

这里笔刷color是外部属性,将由客户端提供,否则Pen的所有内容都将保持不变。 因此,从本质上讲,仅当颜色不同时,我们才会创建一定大小的笔。 一旦其他客户或环境需要该笔的大小和颜色,我们将重新使用它。

  1. import java.util.HashMap;
  2. public class PenFactory
  3. {
  4. private static final HashMap<String, Pen> pensMap = new HashMap<>();
  5. public static Pen getThickPen(String color)
  6. {
  7. String key = color + "-THICK";
  8. Pen pen = pensMap.get(key);
  9. if(pen != null) {
  10. return pen;
  11. } else {
  12. pen = new ThickPen();
  13. pen.setColor(color);
  14. pensMap.put(key, pen);
  15. }
  16. return pen;
  17. }
  18. public static Pen getThinPen(String color)
  19. {
  20. String key = color + "-THIN";
  21. Pen pen = pensMap.get(key);
  22. if(pen != null) {
  23. return pen;
  24. } else {
  25. pen = new ThinPen();
  26. pen.setColor(color);
  27. pensMap.put(key, pen);
  28. }
  29. return pen;
  30. }
  31. public static Pen getMediumPen(String color)
  32. {
  33. String key = color + "-MEDIUM";
  34. Pen pen = pensMap.get(key);
  35. if(pen != null) {
  36. return pen;
  37. } else {
  38. pen = new MediumPen();
  39. pen.setColor(color);
  40. pensMap.put(key, pen);
  41. }
  42. return pen;
  43. }
  44. }

让我们使用客户端来测试重量轻的笔对象。 客户端在这里创建了三支THIN笔,但在运行时它们只是一个瘦型的笔对象,并且与所有三个调用共享。

  1. public class PaintBrushClient
  2. {
  3. public static void main(String[] args)
  4. {
  5. Pen yellowThinPen1 = PenFactory.getThickPen("YELLOW"); //created new pen
  6. yellowThinPen1.draw("Hello World !!");
  7. Pen yellowThinPen2 = PenFactory.getThickPen("YELLOW"); //pen is shared
  8. yellowThinPen2.draw("Hello World !!");
  9. Pen blueThinPen = PenFactory.getThickPen("BLUE"); //created new pen
  10. blueThinPen.draw("Hello World !!");
  11. System.out.println(yellowThinPen1.hashCode());
  12. System.out.println(yellowThinPen2.hashCode());
  13. System.out.println(blueThinPen.hashCode());
  14. }
  15. }

程序输出。

  1. Drawing THICK content in color : YELLOW
  2. Drawing THICK content in color : YELLOW
  3. Drawing THICK content in color : BLUE
  4. 2018699554 //same object
  5. 2018699554 //same object
  6. 1311053135

5.常见问题

5.1 单例模式和享元模式之间的区别

单例模式有助于我们仅在系统中维护一个对象。 换句话说,一旦创建了所需的对象,我们就无法创建更多对象。 我们需要在应用的所有部分中重用现有对象。

当我们必须根据客户端提供的外部属性创建大量不同的相似对象时,将使用享元模式。

5.2 并发对享元的影响

类似于单例模式,如果我们在并发环境中创建享元对象,则最终可能会遇到同一个享元对象的多个实例,这是不希望的。

要解决此问题,我们需要在创建享元时使用单例模式中使用的双检锁

5.3 享元设计模式的好处

使用享元装置,我们可以:

  • 减少可以相同控制的重物的内存消耗。
  • 减少系统中“完整但相似的对象”的总数。
  • 提供了集中机制来控制许多“虚拟”对象的状态。

5.4 内部和外部数据可以共享吗?

内部数据是可共享的,因为它是所有上下文通用的。 不共享外部数据。 客户需要将信息(状态)传递给飞锤,这是其上下文所特有的。

5.5 享元模式的挑战

  • 我们需要花一些时间来配置这些享元。 最初,设计时间和技能可能是开销。
  • 为了创建重量级,我们从现有对象中提取一个通用的模板类。 附加的编程层可能很棘手,有时很难调试和维护。
  • 享元模式通常与单例工厂实现结合使用,并且为了保护奇点,需要额外的成本。

在评论中向我发送有关享元模式的问题。

学习愉快!