一、概述
运用共享技术有效地支持大量细粒度的对象。
二、适用性
1.当一个应用程序使用了大量的对象的时候。
2.由于使用大量的独享而造成很大的存储开销的时候。
3.对象的大多数状态都可变为外部状态的时候。
4.如果删除对象的外部状态,那么可以用相对较少的共享独享取代很对组对象的时候。
5.应用程序不依赖于对象标识的时候。由于享元模式对象可以被共享,对于概念上明显有别的对象,标识测试将返回真值。
三、参与者
1.Flyweight:描述一个接口,通过这个接口Flyweight可以接受和作用于对象外部状态。
2.ConcreteFlyweight:实现Flyweight接口,并为内部状态(如果有)增加存储空间。
3.UnsharedConcreteFlyweight:并非所有的Flyweight子类都需要被共享。Flyweight接口使共享成为可能,但并不强制共享。在Flyweight对象结构的某些层次上,UnsharedConcreteFlyweight对象通常将ConcreteFlyweight对象作为子节点。
4.FlyweightFactory:创建并管理Flyweight对象。确保合理地共享Flyweight对象。当用户请求一个Flyweight对象时,FlyweightFactory提供一个已经创建好的的实例或者创建一个新的实例给它。
四、类图
五、示例
Flyweight
package cn.lynn.flyweight; public interface IPerson { public void say(int num); }ConcreteFlyweight
package cn.lynn.flyweight; public class PersonImpl implements IPerson { @Override public void say(int num) { System.out.println("报数:" + num); } }FlyweightFactory
package cn.lynn.flyweight; import java.util.HashMap; import java.util.Map; public class PersonCache { private static Map<String, PersonImpl> personPool = new HashMap<String, PersonImpl>(); public PersonCache(String key) { personPool.put(key, new PersonImpl()); } public static IPerson getPerson(String key) { if(personPool.get(key) == null) { personPool.put(key, new PersonImpl()); } return personPool.get(key); } public static int getSize() { return personPool.size(); } }Client
package cn.lynn.flyweight; public class Client { public static void main(String[] args) { IPerson p1 = PersonCache.getPerson("张三"); p1.say(1); IPerson p2 = PersonCache.getPerson("张三"); System.out.println("p1和p2是同一个人吗?" + ((p1 == p2) ? "是" : "不是")); IPerson p3 = PersonCache.getPerson("李四"); p3.say(2); IPerson p4 = PersonCache.getPerson("王五"); p4.say(3); IPerson p5 = PersonCache.getPerson("赵六"); p5.say(4); System.out.println("现有人数:" + PersonCache.getSize()); } }
Result
报数:1 p1和p2是同一个人吗?是 报数:2 报数:3 报数:4 现有人数:4