找回密码
 立即注册
首页 业界区 安全 16.结构型 - 享元模式 (Flyweight Pattern)

16.结构型 - 享元模式 (Flyweight Pattern)

莠畅缕 2 小时前
享元模式 (Flyweight Pattern)

享元模式 (flyweight pattern) 的原始定义是:摒弃了在每个对象中保存所有数据的方式,通过共享多个对象所共有的相同状态,从而让我们能在有限的内存容量中载入更多对象。
享元对象分为两类数据:

  • 内部状态: 不可被改变, 可共享
  • 外部状态: 可被改变, 不可共享
例如: 生活中的一个共享单车例子,无论是谁在使用/占用,单车本体:轮子、车架等都是一样,不会改变,这些数据是(内部状态);而是谁在使用/占用,以及使用的时长,费用等是根据具体人而定的,这些数据是(外部状态)
从这个定义中你可以发现,享元模式要解决的核心问题就是节约内存空间,使用的办法是找出相似对象之间的共有特征,然后复用这些特征。所谓“享元”,顾名思义就是被共享的单元。
UML 类图

1.png

代码实例

1.抽象享元对象 (共享单车)
  1. /**  
  2. * 抽象享元对象  
  3. */
  4. abstract class BikeFlyWeight{
  5.         int state;//使用状态 1=使用中,0未使用 内部状态
  6.         //使用
  7.         abstract void use(String username);
  8.         //归还
  9.         abstract void back();
  10.         //返回使用状态
  11.         public int getState(){
  12.                 return this.state;
  13.         }
  14. }
复制代码
2.具体享元对象 (摩拜单车)
  1. /**
  2. * 具体享元对象
  3. */
  4. class MoBikeFlyWeight extends BikeFlyWeight{
  5.         private String bikeId;
  6.         private String username;
  7.         public MoBikeFlyWeight(String bikeId){
  8.                 this.bikeId = bikeId;
  9.         }
  10.         @Override
  11.         void use(String username) {
  12.                 this.username = username;
  13.                 System.out.println(bikeId+" 号单车, "+ username + ", 使用中");
  14.                 state = 1;
  15.         }
  16.         @Override
  17.         void back() {
  18.                 System.out.println(bikeId+" 号单车, "+ username + ", 已归还");
  19.                 state = 1;
  20.         }
  21. }
复制代码
3.享元工厂 (缓冲池)

它充当一个池子的作用, 客户端从它这里获取享元对象, 以便复用对象;
  1. public class BikeFlyWeightFactory {
  2.     private static BikeFlyWeightFactory instance = new BikeFlyWeightFactory();
  3.     private static BikeFlyWeightFactory getInstance(){return instance; }
  4.     private Set<BikeFlyWeight> pool = new HashSet<>();
  5.     private BikeFlyWeightFactory(){
  6.         //初始化缓存池
  7.         for (int i = 0; i < 5; i++) {
  8.             pool.add( new MoBikeFlyWeight("mo-"+i ));
  9.         }
  10.     }
  11.         public BikeFlyWeight getBike(){
  12.         for (Iterator<BikeFlyWeight> iterator = pool.iterator(); iterator.hasNext(); ) {
  13.             BikeFlyWeight next =  iterator.next();
  14.             if (next.getState() == 0 ){
  15.                 return next;
  16.             }
  17.         }
  18.         return null;
  19.     }
  20. }
复制代码
测试
  1. public static void main(String[] args) {
  2.         BikeFlyWeightFactory instance = getInstance();
  3.        
  4.         BikeFlyWeight bike = instance.getBike();
  5.         bike.use("张三");
  6.         BikeFlyWeight bike2 = instance.getBike();
  7.         bike2.use("李四");
  8.         bike.back();//张三归还
  9.         BikeFlyWeight bike3 = instance.getBike();//张三归还了, 获取到 1 号对象, 复用了资源
  10.         bike3.use("王五");
  11. }
  12. // out
  13. // mo-1 号单车, 张三, 使用中  
  14. // mo-2 号单车, 李四, 使用中  
  15. // mo-1 号单车, 张三, 已归还  
  16. // mo-1 号单车, 王五, 使用中
复制代码
享元模式 (Flyweight Pattern) 总结

享元模式通过共享技术实现相同或者相似对象的重用,在逻辑上每一个出现的字符都有一个对象与之对应,然而在物理上他们却是共享同一个享元对象.
关键角色


  • 抽象享元对象
  • 具体享元对象
  • 享元工厂
享元模式的优缺点

享元模式的优点

  • 极大减少内存中相似或相同对象数量,节约系统资源,提供系统性能
    比如,当大量商家的商品图片、固定文字(如商品介绍、商品属性)在不同的网页进行展示时,通常不需要重复创建对象,而是可以使用同一个对象,以避免重复存储而浪费内存空间。由于通过享元模式构建的对象是共享的,所以当程序在运行时不仅不用重复创建,还能减少程序与操作系统的 IO 交互次数,大大提升了读写性能。

  • 享元模式中的外部状态相对独立,且不影响内部状态
享元模式的缺点

  • 为了使对象可以共享,需要将享元对象的部分状态外部化,分离内部状态和外部状态,使程序逻辑复杂
享元模式的适用场景


  • 一个系统有大量相同或者相似的对象,造成内存的大量耗费。
    注意: 在使用享元模式时需要维护一个存储享元对象的享元池,而这需要耗费一定的系统资源,因此,应当在需要多次重复使用享元对象时才值得使用享元模式。
  • 在 Java 中,享元模式一个常用的场景就是,使用数据类的包装类对象的 valueOf() 方法。比如,使用 Integer.valueOf() 方法时,实际的代码实现中有一个叫 IntegerCache 的静态类,它就是一直缓存了 -127 到 128 范围内的数值,如下代码所示,你可以在 Java JDK 中的 Integer 类的源码中找到这段代码。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册