享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能。这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的方式。
主要解决:
在有大量对象时,有可能会造成内存溢出,我们把其中共同的部分抽象出来,如果有相同的业务请求,直接返回在内存中已有的对象,避免重新创建。何时使用:
1、系统中有大量对象。2、这些对象消耗大量内存。3、这些对象的状态大部分可以外部化。4、这些对象可以按照内蕴状态分为很多组,当把外蕴对象从对象中剔除出来时,每一组对象都可以用一个对象来代替。5、系统不依赖于这些对象身份,这些对象是不可分辨的。如何解决:
用唯一标识码判断,如果在内存中有,则返回这个唯一标识码所标识的对象。关键代码:
用 HashMap 存储这些对象。应用实例 :
1、JAVA 中的 String,如果有则返回,如果没有则创建一个字符串保存在字符串缓存池里面。2、数据库的数据池。优点
大大减少对象的创建,降低系统的内存,使效率提高。缺点:
提高了系统的复杂度,需要分离出外部状态和内部状态,而且外部状态具有固有化的性质,不应该随着内部状态的变化而变化,否则会造成系统的混乱。使用场景 :
1、系统有大量相似对象。2、需要缓冲池的场景。注意事项:
1、注意划分外部状态和内部状态,否则可能会引起线程安全问题。2、这些类必须有一个工厂对象加以控制。用户使用网站的案例:一个用户使用一个网站需要先创建网站,如果之前有人已经使用创建了一个网站,就共享同一个网站
先创建用户类
/** * 用户类 * * @author cwh */ public class User { private String name; public User(String name) { super(); this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; } }创建抽象网站类
/** * @author cwh */ public abstract class AbstractWebSite { /** * 抽象方法,接收使用用户 * * @param user */ public abstract void use(User user); }创建具体的网站
/** * 具体网站 * * @author cwh */ public class ConcreteAbstractWebSite extends AbstractWebSite { /** * 网站发布的形式(类型) */ private String type; public ConcreteAbstractWebSite(String type) { this.type = type; } @Override public void use(User user) { System.out.println("网站的发布形式为:" + type + " 在使用中 .. 使用者是" + user.getName()); } }用一个简单工厂来实现共享网站
/** * 简单的网站工厂 * * @author cwh */ public class WebSiteFactory { /** * 集合, 充当池的作用 */ private static HashMap<String, ConcreteAbstractWebSite> webSitePool = new HashMap<>(); /** * 根据网站的类型,返回一个网站, 如果没有就创建一个网站,并放入到池中,并返回 * * @param type * @return */ public static AbstractWebSite getWebSiteCategory(String type) { if (!webSitePool.containsKey(type)) { webSitePool.put(type, new ConcreteAbstractWebSite(type)); } return webSitePool.get(type); } /** * 获取网站分类的总数 (池中有多少个网站类型) * * @return */ public static int getWebSiteCount() { return webSitePool.size(); } }客户端
public class Client { public static void main(String[] args) { //客户要一个以新闻形式发布的网站 AbstractWebSite abstractWebSite1 = WebSiteFactory.getWebSiteCategory("新闻"); abstractWebSite1.use(new User("tom")); //客户要一个以博客形式发布的网站 AbstractWebSite abstractWebSite2 = WebSiteFactory.getWebSiteCategory("博客"); abstractWebSite2.use(new User("jack")); //客户要一个以博客形式发布的网站 AbstractWebSite abstractWebSite3 = WebSiteFactory.getWebSiteCategory("博客"); abstractWebSite3.use(new User("smith")); //客户要一个以博客形式发布的网站 AbstractWebSite abstractWebSite4 = WebSiteFactory.getWebSiteCategory("博客"); abstractWebSite4.use(new User("king")); System.out.println("网站的分类共=" + WebSiteFactory.getWebSiteCount()); } }