原型 和 单例模式

2019-02-28  本文已影响0人  NazgulSun

说实话,这两个模式已经用的比较少了。

先说单例模式,由于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,这种方式比较麻烦;

另外一个就是使用序列化机制,把当前兑现写入到 字节流里面,再从字节流里读出来创建新对象。

上一篇下一篇

猜你喜欢

热点阅读