java设计模式---享元模式

享元模式

享元模式的定义

使用共享对象可有效地支持大量细粒度对象

为什么要共享对象

因为分配太多的对象到应用程序中将有损程序的性能,同时还容易造成内存溢出.

java中的内存泄漏原因:

  • 无意识的代码缺陷,导致内存泄露,jvm不能获得连续的内存空间
  • 产生的对象过多,导致被耗尽

什么是内部状态和外部状态

1.内部状态是对象可共享出来的信息,存储在享元对象的内部并且不会随环境的改变而改变.他们可以作为一个对象的动态附加信息,
不必直接存储在具体的某个对象中,属于可以共享的部分.
2.外部状态是对象得以依赖的一个标记,是随环境的改变而改变的,不可以共享的状态,他是一批对象的统一标识,是唯一的一个索引值.

享元模式的角色

  • Flyweight 抽象享元角色
  • ConcreteFlyweight 具体享元角色
  • unConcreteFlyweight 不可共享的享元角色
  • FlyweightFactory 享元工厂

具体类图如下

这里写图片描述

实现示例

假如有个考试报名的信息类,考试科目和考试地点是确定的,为外部信息,个人的id是不确定的,由具体对象确定,为内部对象,现在要创建
很多这样的对象,用到享元模式.

报考信息

/**
 * 
 *  报考信息类
 *
 */
public class SignInfo {
    // id
    private String id;
    // 地点
    private String location;
    // 科目
    private String subject;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getLocation() {
        return location;
    }

    public void setLocation(String location) {
        this.location = location;
    }

    public String getSubject() {
        return subject;
    }

    public void setSubject(String subject) {
        this.subject = subject;
    }

}

带对象池的报考信息类

/**
 * 
 * 用于对象池中存储对象
 * 
 */
public class SignInfo4Pool extends SignInfo {

//用于存储的key
private String mKey;

public SignInfo4Pool(String key) {
    this.mKey = key;
}

public String getmKey() {
    return mKey;
}

public void setmKey(String mKey) {
    this.mKey = mKey;
}

}

创建报考信息的工厂类

/**
 * 
 * 创建对象的工厂
 * 
 */
public class SignInfoFactory {

    //对象缓冲池
    private static HashMap<String, SignInfo> pool = new HashMap<String, SignInfo>();

    /**
     *获取signInfo
     */
    public static SignInfo getSignInfo(String key) {
        SignInfo signInfo = null;
        if (!pool.containsKey(key)) {
            signInfo = new SignInfo4Pool(key);
            pool.put(key, signInfo);
        } else {
            signInfo = pool.get(key);
        }
        return signInfo;
    }

}

客户端类

 public class Client {

    public static void main(String[] args) {
        // 初始化科目
        for (int i = 0; i < 4; i++) {
            // 构建key
            String subject = "科目" + i;
            // 初始化地址
            for (int j = 0; j < 30; j++) {
   String key = subject + "地址" + j;
   SignInfoFactory.getSignInfo(key);
            }
        }
        SignInfo signInfo = SignInfoFactory.getSignInfo("科目1考试地点1");
    }

}

享元模式的优点和缺点

可以大大减少应用程序创建的对象,降低程序内存的占用,增强程序的性能.但是它也提高了系统的复杂性,而且需要分离出内部状态和
外部状态.而且外部状态具有固化性.

应用场景

  • 系统中存在大量的相似对象.
  • 细粒度的对象都具备较接近的外部状态,而且内部状态与环境无关,也就是说对象没有特定身份.
  • 需要缓冲池的场景.

需要注意的地方

  • 享元模式容易造成线程安全的问题,所以尽可能的创建多的对象
  • 享元模式存储在对象缓冲池中的key尽量用int,string等基本数剧类型作为key,而不是具体的对象,这样可以显著提高性能.
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值