Singleton单例

前言

以前我自认为对设计模式较为熟悉了,最起码最简单的单例模式应该已经很会了。可是,我真的会了吗?书只会越读越薄,每一次的回顾都带来新的发现。

第一种写法

这是最简单的一种写法,简单实用,推荐使用!

public class Mgr01 {
    private static final Mgr01 INSTANCE = new Mgr01();
    private Mgr01(){};
    public static Mgr01 getInstance(){
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        Mgr01 m1 = Mgr01.getInstance();
        Mgr01 m2 = Mgr01.getInstance();
        System.out.println(m1 == m2);
    }
}

唯一缺点:不管用到与否,类加载时就完成实例化

第二种写法

这个其实跟第一种写法意思一样的,只不过是把初始化放在静态语句块里了。

public class Mgr02 {
    private static final Mgr02 INSTANCE;
    static {
        INSTANCE = new Mgr02();
    }
    private Mgr02(){};
    public static Mgr02 getInstance(){
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        Mgr02 m1 = Mgr02.getInstance();
        Mgr02 m2 = Mgr02.getInstance();
        System.out.println(m1 == m2);
    }
}

第三种写法

该写法为懒汉式,虽然达到了按需初始化的目的,但却带来了线程不安全的问题。

public class Mgr03 {
    private static Mgr03 INSTANCE;
    private Mgr03(){};
    public static Mgr03 getInstance(){
        if (INSTANCE == null){
            INSTANCE = new Mgr03();
        }
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(
                        Mgr03.getInstance().hashCode()
                );
            }).start();
        }
    }
}

第四种写法

上面第三种写法可以通过synchronized解决,但也带来效率低下。

public class Mgr04 {
    private static Mgr04 INSTANCE;
    private Mgr04(){};
    public static synchronized Mgr04 getInstance(){
        if (INSTANCE == null){
            INSTANCE = new Mgr04();
        }
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Mgr04.getInstance().hashCode());
            }).start();
        }
    }
}

第五种写法

对于上面的第四种写法,妄图通过减少同步代码块的方式见提高效率,然后不可行。

public class Mgr05 {
    private static Mgr05 INSTANCE;
    private Mgr05(){};
    public static  Mgr05 getInstance(){
        if (INSTANCE == null){
            synchronized(Mgr05.class){
                INSTANCE = new Mgr05();
            }
        }
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Mgr05.getInstance().hashCode());
            }).start();
        }
    }
}

第六种写法

在以前,这种写法是完美的一种写法,采用双重if判断。

public class Mgr06 {
    private static Mgr06 INSTANCE;
    private Mgr06(){
    }
    public static Mgr06 getInstance(){
        if (INSTANCE == null){
            synchronized (Mgr06.class){
                if (INSTANCE == null){
                    INSTANCE = new Mgr06();
                }
            }
        }
        return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Mgr06.getInstance().hashCode());
            }).start();
        }
    }
}

第七种写法

其实上面的第六种写法还不够完美,有一点小瑕疵,第七种写法采用了静态内部类方式,由JVM保证单例。这是最完美的两种写法之一,为什么说是之一,因为第八种也是最完美的写法之一。

public class Mgr07 {
    private Mgr07(){};
    private static class Mgr07Holder{
        private final static Mgr07 INSTANCE = new Mgr07();
    }
    public static Mgr07 getInstance(){
        return Mgr07Holder.INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Mgr06.getInstance().hashCode());
            }).start();
        }
    }
}

第八种写法

这种写法出自《Effective Java》一书,作者亲自写出了一种最简单也最明了的写法,采用了枚举单例,这是八种写法中唯一一种可以避免反序列化的写法。同时也解决了线程同步。

public enum  Mgr08 {
    INSTANCE;
    public void m(){}

    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Mgr08.INSTANCE.hashCode());
            }).start();
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

༄༊心灵骇客༣

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值