享元模式

来源:互联网 发布:jni 调用java实例方法 编辑:程序博客网 时间:2024/06/04 18:02

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

起先没看懂为什么享元模式要那么写,就知道享元可以避免大量相似类的开销。就是说要生成大量细粒度的数据,比如说String就用了享元,这些细粒度的数据除了几个参数外基本是相同的,用享元将这些参数移到类的实例外,在方法调用时将它们传递进来,能够大幅度的减少单个实例的数目。

那么现在来看看享元模式到底是怎么做到的。比如说我们要建网站,网站类分为展示网站和博客,它们肯定有很多相似的地方。

先抽象一个网站类,这里为什么要有一个网站抽象类而不是直接是一个具体网站类,现在想明白了,是因为享元以具体网站类实例化后的对象为类型,比如说这里我们会动态创建 展示和博客 两个对象,如果还要创建博客对象就直接用前面创建的博客对象就可以了,不会再new一个博客对象了,用博客对象的方法将用户信息传给对象,所以这里有具体网站类和抽象网站类的划分,具体网站类实例化的对象定义了网站类型,就需要一个抽象类来一起调用这些网站类的方法,我们不可能在去一个个用具体网站类去调用方法吧。

public interface WebSite {    void use(User user);}
public class ConcreteWebSite implements WebSite {    private String name = "";    public ConcreteWebSite(String name) {        this.name = name;    }    @Override    public void use(User user) {        System.out.println("网站分类:" + name + " 用户:" + user.getName());    }}

这个User就是外部数据,我们把它放到了具体网站类实例化外,通过具体网站类的方法来调用。

public class User {    private String name;    public User(String name) {        this.name =name;    }    public String getName() {        return name;    }}

享元最重要的就是相似的对象只需要实例化一次,这里我们维护一张hashtable的表,为什么用hashtable而不用hashmap是因为hashtable是多线程安全的,这里我们需要任何情况相同的网站类都只实例化一次,所以要保证多线程情况下也只实例化一次。

public class WebSiteFactory {    private Hashtable<String, WebSite> webSites = new Hashtable<String, WebSite>();    public WebSite getWebSiteCategory(String key) {        if(!webSites.containsKey(key)) {            webSites.put(key, new ConcreteWebSite(key));        }        return (WebSite) webSites.get(key);    }    public int getWebSiteCount() {        return webSites.size();    }}

这就是享元模式如何实现的原理,这个函数相信都能看懂,利用工厂模式,用getWebSiteCategory来保证只有key不同才会实例化新的具体网站。

主流程代码:

        WebSiteFactory f= new WebSiteFactory();        WebSite fx = f.getWebSiteCategory("产品展示");        fx.use(new flyweight.User("小明"));        WebSite fy = f.getWebSiteCategory("产品展示");        fy.use(new flyweight.User("小花"));        WebSite fz = f.getWebSiteCategory("产品展示");        fz.use(new flyweight.User("小李"));        WebSite fl = f.getWebSiteCategory("博客");        fl.use(new flyweight.User("小强"));        WebSite fm = f.getWebSiteCategory("博客");        fm.use(new flyweight.User("小张"));        WebSite fn = f.getWebSiteCategory("博客");        fn.use(new flyweight.User("小王"));        System.out.println("网站分类总数:" + f.getWebSiteCount());

运行结果:

网站分类:产品展示 用户:小明网站分类:产品展示 用户:小花网站分类:产品展示 用户:小李网站分类:博客 用户:小强网站分类:博客 用户:小张网站分类:博客 用户:小王网站分类总数:2

看运行结果,我们虽然建立了6个网站,但它们其实公用了两个对象,用户信息作为外部参数传入,通过抽象接口统一调用,这就是我理解的享元模式的原理。

0 0
原创粉丝点击