Java设计模式-单例模式

简介

当有些对象我们只需要一个,例如:配置文件,工具类,线程池,缓存,日志对象等。如果这时候创建出多个对象就会导致内存占用过多,不一致的结果等等。这时候就可以通过单例模式来实现让对象只有一个。

模式的分类

-饿汉模式
特点:加载类的时候比较慢,但是在运行时获取类的实例比较快,线程安全。

public class Singleton {

    //在类加载的时候就new一个当前类的实例
    private static Singleton instance = new Singleton();

    //私有化构造方法,不让外界创建实例
    private Singleton() {

    }

    //对外界提供一个获取实例的静态方法
    public static Singleton getInstance() {
        return instance;
    }

}

-懒汉模式:
特点:加载类的时候比较快,但是在运行时获取类的实例比较慢,线程不安全。

public class Singleton {

    //只是声明,在类加载的时候并不实例化当前类
    private static Singleton instance;

    //私有化构造方法,不让外界创建实例
    private Singleton() {

    }

    //对外界提供一个获取实例的静态方法
    public static Singleton getInstance() {
        if (instance == null) {
            //在外界需要获取当前类实例的时候才new一个出来
            instance = new Singleton();
        }
        return instance;
    }
}

-为什么懒汉模式线程不安全:

在多线程运行情况下,如果有两个线程同时在执行getInstance方法,第一个线程刚执行完if语句的判断,还没执行到if语句块里面,这个时候第二个线程执行到了if语句判断中,它会发现instance还是等于null,于是进入到了if语句块里面。这样第二个线程创建了一个当前实例,而当第一个线程又开始执行的时候,因为它之前已经经过if语句判断了,所以不会再次判断instance是否为null就直接进入到了if语句块中,于是第一个线程又创建了一个实例。在这种情况下就会导致有两个实例被创建,单例模式失败,严格意义上来说,以上这种方式并不能算是真正的单例模式。

-如何解决:

方法很简单,给获取实例的getInstance方法加上一个同步锁(synchronized)就可以了。

public class Singleton {

    //在类加载的时候并不实例化当前类
    private static Singleton instance;

    //私有化构造方法,不让外界创建实例
    private Singleton() {

    }

    //对外界提供一个获取实例的方法(加同步锁)
    public synchronized static Singleton getInstance() {
        if (instance == null) {
            //在外界需要获取当前类实例的时候才new一个出来
            instance = new Singleton();
        }
        return instance;
    }
}

以上这种方式虽然解决了在多线程运行情况中创建出不止一个实例的问题,但是在运行速度上还存在可以优化的空间。原因在于getInstance方法上加了一个synchronized同步锁,那么每次去执行getInstance方法的时候都会受到同步锁的影响,这样运行的效率就会降低,其实只要在第一次创建instance实例的时候加上同步锁就好了。

-优化方式:

public class Singleton {

    //在类加载的时候并不实例化当前类
    private static Singleton instance;

    //私有化构造方法,不让外界创建实例
    private Singleton() {

    }

    //对外界提供一个获取实例的静态方法
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    //在外界需要获取当前类实例的时候才new一个出来
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

在这里为什么需要加两个判断呢?第一个判断是优化运行效率的关键所在,当instance不为null的时候直接返回当前类的实例,无需进入同步块中。当instance为null的时候就进入同步块中执行第二个判断,作用与前文一样。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值