Juconcurrent 学而不思则罔,思而不学则殆。

单例模式


概念

所谓单例,就是从头到尾只有一个对象的意思。不过,为什么只能有一个对象呢?我觉得可能有以下原因:

  1. 对象本身是线程安全的,不需要多个对象;
  2. 创建多个对象的开销比较大;

单例主要的使用场景在于组件化,比如:spring的bean管理、工具类。

另外,有时候我们可以从单例扩展到多例,这种情况比较常见的场景就是对象池。比如:数据库连接池、线程池等。

示例1,最简单的单例

public class Single {

    private static final Single instance = new Single();

    public static Single instance() {
        return instance;
    }
}

示例2,枚举型单例

这种方式能避免反序列化和反射方式创建多个单例的问题。我这儿最推荐这种方式,虽然以枚举的方式实现比较变态

public enum Single {

    single;

    public void calc() {}
}

示例3,延迟初始化的单例

这儿为啥需要加锁,以及锁里面为啥需要再判断一次null呢?这是因为多线程里面可以使用锁来保证内存可见性和顺序性。 而至于为啥需要在锁里面再判断一次空指针,是因为被阻塞线程再阻塞线程执行完后,看见的instance已经是非null的,但是没有条件能退出实例化。

public class Single {

    private static Single instance;

    public static Single instance() {
        if (instance == null) {
            synchronized (Single.class) {
                if (instance == null) {
                    instance = new Single();
                }
            }

        }
        return instance;
    }
}

示例4,另外一种延迟初始化的单例

使用内部类加载机制来实现延迟初始化,弊端就是会额外多维护一个类,增加了维护的成本。

public class Single {

    public static Single instance() {
        return Holder.single;
    }

    private static class Holder {

        private static final Single single = new Single();
    }
}

下一篇 模板模式

Content