原型 和 单例模式
说实话,这两个模式已经用的比较少了。
先说单例模式,由于spring 的大量使用,基本上通过 spring 来创建对象,在compoment上指定 是单例还是多例模式已经比较方便。
不过作为面试,单例模式还是经常被问道,通常是考虑的在多线程下如何确保单例的。
if(singleton ==null) {
synchronized(Singleton.class) {
if(singleton ==null) {
singleton =new Singleton();
}
}
}
这种写法是有问题的, 为啥? 问题出在了 singleton = new Singleton 这句话上。
java 不保证 是先 new 再 赋值。 由于优化需要,可能是先分配 内存 然后赋值, 这可能导致别人用的是一个 没有初始化的 instance。
这个点是非常的隐蔽,涉及到java 指令优化 从新排列。
所以推荐的写法 是使用 静态内部类,由java classloader 负责实例化和赋值,这个是jvm 虚拟机规范保障的。
private Singleton() {}
private static class SingletonInstance {
private static final Singleton INSTANCE =new Singleton();
}
public static Singleton getInstance() {
return SingletonInstance.INSTANCE;
}
其实在 effective jvm 里面还谈到了使用enum 来做单例,也是利用jvm 保障 enum声明的instance之外不能创建额外的instance。
为了保障唯一性,通常还需要考虑 不让 singleinstance 实现 序列化,因为序列化是可以绕开 构造函数的,在singleinstance 里面重写read 实例化方法就可以确保万无一失。
对于原型模式,实际的编程中用的很少,不过扯出来的相应的知识点还是比较多。
第一:主要是clone方法。如果对象想支持clone,必须实现一个 clonable 声明接口,该声明接口在java中没有任何方法。
java 一共有三个申明接口。 RandomAccess,Clonable, Serieliable
对于实现clonable接口的对象,可以调用 clone方法
第二: clone 方法涉及到深拷贝和前拷贝,对于基本类型和string 都是在造一个一模一样的数据,对于引用对象,浅拷贝只是拷贝引用,
完成深拷贝的方法大概有两种, 对于引用对象 也递归实现clone,这种方式比较麻烦;
另外一个就是使用序列化机制,把当前兑现写入到 字节流里面,再从字节流里读出来创建新对象。