设计模式-享元模式

它使用物件用来尽可能减少内存使用量;于相似物件中分享尽可能多的资讯。当大量物件近乎重复方式存在,因而使用大量内存时,此法适用。通常物件中的部分状态(state)能够共享。常见做法是把它们放在数据结构外部,当需要使用时再将它们传递给享元。

享元模式又称为轻量级模式,是对象池的一种实现。类似于线程池可以避免不停的创建和销毁多个对象,消耗性能。

它的本质是通过共享对象(将多个同一对象的访问集中起来)来降低内存消耗,是结构型模式。

这里需要注意的是享元模式把一个对象的状态分为内部状态和外部状态,内部状态是不变的可以共享的相同内容,外部状态是变化的是需要外部环境来设置的不能共享的内容,需要注意内部状态和外部状态的区分

模式的动机

主要是为了解决当对象数量太多时,将导致运行代价过高,带来性能下降等问题,通常境况下底层可能会用的更多些,比如我们的源码底层,String、Integer、Long等下文会说到

模式的结构

享元模式的通用UML类图

通用代码:

public interface Flyweight {
    void operation(String extrinsicState);
}
复制代码
public class ConcreteFlyweight implements Flyweight{
    private String intrinsicState;

    public ConcreteFlyweight(String intrinsicState){
        this.intrinsicState = intrinsicState;
    }

    @Override
    public void operation(String extrinsicState) {
        System.out.println("extrinsicState = " + extrinsicState);
        System.out.println("intrinsicState = " + this.intrinsicState);
    }
}
复制代码

这里需要注意的点事可能有的人会觉得intrinsicState这个属性是外部传过来的所以他就是外部对象,其实不然,工厂就是通过intrinsicState来区分实例的,而且也只有构造方法会用到其他是没有方法能够改变这个属性的,所以把这个属性定义为内部属性也就是可以共享的

下面的operation方法就会接受一个外部属性,这个是不共享的所以定义为外部属性

一定一定要区分外部属性和内部属性

public class FlyweightFactory {
    private static Map<String,Flyweight> pool = new HashMap<>();

    // 因为内部状态具备不变性,因此作为缓存的键
    public static Flyweight getFlyweight(String intrinsicState) {
        if (!pool.containsKey(intrinsicState)) {
            Flyweight flyweight = new ConcreteFlyweight(intrinsicState);
            pool.put(intrinsicState, flyweight);
        }
        return pool.get(intrinsicState);
    }
}
复制代码
public class Test {
    public static void main(String[] args) {
        Flyweight flyweight1 = FlyweightFactory.getFlyweight("aa");
        Flyweight flyweight2 = FlyweightFactory.getFlyweight("bb");
        flyweight1.operation("a");
        flyweight2.operation("b");
    }
}
复制代码

代码示例

通过上文的解说可能有些人对内部状态和外部状态还是很模糊,也不知道到底什么时候去使用享元模式,这里我们再举一个例子,比如我们下五子棋,所有棋大小形状肯定都是一样的,我们可以作为内部状态共享,其实我们还可以把颜色作为内部状态共享,有的人可能角色颜色有两种怎么共享。这里我们可以将颜色通过构造方法传入赋值给颜色属性,在工厂创建实例的时候就把白色和黑色划分成了两种对象,每一种对象的颜色就统一了。外部状态就是棋的坐标。

具体实现如下:

public abstract class BaseChess {
    //这两个属性是所有棋子共有的属性所以我们一颗直接放到抽象类中
    protected final String shape = "原型";
    protected final Integer radius = 5;
    protected String color;

    public BaseChess(String color){
        this.color = color;
    }
    //移动棋子
    public abstract void moveChess(int x, int y);
}
复制代码
public class Chess extends BaseChess {

    public Chess(String color) {
        super(color);
    }

    @Override
    public void moveChess(int x, int y) {
        String string = "棋子形状:" +
                super.shape +
                "棋子半径:" +
                super.shape +
                "----" +
                super.color +
                "棋子移动位置" +
                "x:" +
                x +
                "y:" +
                y;
        System.out.println(string);
    }
}
复制代码
public class ChessFactory {
    private static Map<String,BaseChess> pool = new HashMap<>();

    public static BaseChess getChess(String color){
        //这里以棋子颜色作为键值来区分示例
        if(pool.containsKey(color)){
            return pool.get(color);
        }else{
            Chess chess = new Chess(color);
            pool.put(color,chess);
            return chess;
        }
    }
}
复制代码
public class Test {
    public static void main(String[] args) {
        BaseChess chess = ChessFactory.getChess("白色");
        BaseChess chess1 = ChessFactory.getChess("黑色");
        BaseChess chess2 = ChessFactory.getChess("白色");
        BaseChess chess3 = ChessFactory.getChess("黑色");
        System.out.println(chess);
        chess.moveChess(100,102);
        System.out.println(chess1);
        chess1.moveChess(101,102);
        System.out.println(chess2);
        chess2.moveChess(102,102);
        System.out.println(chess3);
        chess3.moveChess(103,102);
    }
}
复制代码

这里有一点是如果有人觉得这个内部对象还要通过构造方法传递很不爽,可以创建两个对象分别是白色棋子对象和黑色棋子对象,在工厂方法中就定义号白色对应的是哪个对象黑色是哪个,这样就可以将颜色这个内部属性写死在对象内部了

源码中的提现

String

Java 中将 String 类定义为 final(不可改变的),JVM 中字符串一般保存在字符串常量池中,java 会确保一个宁符串在常量池中只有一个拷贝,这个宁符串常量池在JDK6.0 以前是位于常量池中,位于 永久代,而在JDK7.0中,JVM将其从永久代拿出来放置于堆中。

可以看一下下面代码:

String s1 = "hello";
String s2 = "hello";
String s3 = new String("hello");
String s4 = s3.intern();
System.out.println(s1 == s2);
System.out.println(s1 == s3);
System.out.println(s2 == s3);
System.out.println(s4 == s3);
System.out.println(s4 == s2);
复制代码

运行结果如下:

  1. s1和s2一样是应为"hello"这个字面量已经存到常量池中
  2. s2再赋值的时候会到常量池中找找到了直接赋值不会再新创建对象s2和s3不一样就是s3自己new了一个对象所以肯定是不一样的
  3. s4和s3不一样又和s2一样的原因是 intern方法会去查常量池查如果查到了直接返回常量池的对象,如果没查到会将s3放到常量池(通过在创建一个新对象)再返回对应常量池的引用

Integer

可以先看一下 valueOf方法源码:

可以看到会先判断范围如果是在达到一定的范围就直接冲缓存中取否则才会创建对象
再看这个IntegeCache是个什么东西:

private static class IntegerCache {
    static final int low = -128;
    static final int high;
    static final Integer[] cache;
    static Integer[] archivedCache;

    static {
        // high value may be configured by property
        int h = 127;
        String integerCacheHighPropValue =
            VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
        if (integerCacheHighPropValue != null) {
            try {
                h = Math.max(parseInt(integerCacheHighPropValue), 127);
                // Maximum array size is Integer.MAX_VALUE
                h = Math.min(h, Integer.MAX_VALUE - (-low) -1);
            } catch( NumberFormatException nfe) {
                // If the property cannot be parsed into an int, ignore it.
            }
        }
        high = h;

        // Load IntegerCache.archivedCache from archive, if possible
        CDS.initializeFromArchive(IntegerCache.class);
        int size = (high - low) + 1;

        // Use the archived cache if it exists and is large enough
        if (archivedCache == null || size > archivedCache.length) {
            Integer[] c = new Integer[size];
            int j = low;
            for(int i = 0; i < c.length; i++) {
                c[i] = new Integer(j++);
            }
            archivedCache = c;
        }
        cache = archivedCache;
        // range [-128, 127] must be interned (JLS7 5.1.7)
        assert IntegerCache.high >= 127;
    }

    private IntegerCache() {}
}
复制代码

发现是Integer的一个内部类,在第一次调用它的时候就会创建值为-128~127的对象。

享元模式优缺点

优点:

  1. 如果有大量相同对象要创建该模式可以提高性能节省内存

缺点

  1. 享元模式使得系统更加复杂,需要分离出内部状态和外部状态,这使得程序的逻辑复杂化。
  2. 为了使对象可以共享,享元模式需要将享元对象的状态外部化,而读取外部状态使得运行时间变
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值