单例模式总结

概述

定义: 一个类只能产生一个实例,并且该类提供一个唯一的全局访问点。

说明: 在设计系统时,有的对象,需要保证全局只有一个。例如如下对象(线程池,log对象, 注册表信息对象,性能设置对象,驱动类对象,打印机对象等 )

基本思路:

  1. 私有构造器。保证其他类无法实例化该类。
  2. 静态方法。通过该静态方法能得到该对象的实例。

单例模式的应用

单例模式的优点

由于单例模式在内存中只有一个实例,减少了内存开支,特别是一个对象需要频繁地创建、销毁时,而且创建或销毁时性能又无法优化,单例模式的优势就非常明显。由于单例模式只生成一个实例,所以减少了系统的性能开销,当一个对象的产生需要比较多的资源时,如读取配置、产生其他依赖对象时,则可以通过在应用启动时直接产生一个单例对象,然后用永久驻留内存的方式来解决(在Java EE中采用单例模式时需要注意JVM垃圾回收机制)。
单例模式可以避免对资源的多重占用,例如一个写文件动作,由于只有一个实例存在内存中,避免对同一个资源文件的同时写操作。
单例模式可以在系统设置全局的访问点,优化和共享资源访问,例如可以设计一个单例类,负责所有数据表的映射处理。

单例模式的缺点

单例模式一般没有接口,扩展很困难。
若要扩展,除了修改代码基本上没有第二种途径可以实现。单例模式为什么不能增加接口呢?因为接口对单例模式是没有任何意义的,它要求“自行实例化”,并且提供单一实例、接口或抽象类是不可能被实例化的。当然,在特殊情况下,单例模式可以实现接口、被继承等,需要在系统开发中根据环境判断。
单例模式对测试是不利的。在并行开发环境中,如果单例模式没有完成,是不能进行测试的,没有接口也不能使用mock的方式虚拟一个对象。
单例模式与单一职责原则有冲突。一个类应该只实现一个逻辑,而不关心它是否是单例的,是不是要单例取决于环境,单例模式把“要单例”和业务逻辑融合在一个类中。

单例模式的使用场景

在一个系统中,要求一个类有且仅有一个对象,如果出现多个对象就会出现“不良反应”,可以采用单例模式。
具体的场景如下:

  • 要求生成唯一序列号的环境;
  • 在整个项目中需要一个共享访问点或共享数据,例如一个Web页面上的计数器,可以不用把每次刷新都记录到数据库中,使用单例模式保持计数器的值,并确保是线程安全的;
  • 创建一个对象需要消耗的资源过多,如要访问IO和数据库等资源;
  • 需要定义大量的静态常量和静态方法(如工具类)的环境,可以采用单例模式(当然,也可以直接声明为static的方式)。

单例模式的注意事项

在高并发情况下,请注意单例模式的线程同步问题。

Java中有以下几种写法:

一. 懒汉(线程不安全)

public class Singleton {

    private static Singleton instance;

    private Singleton() {

    }

    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

总结: 这种写法起到了lazy loading的效果,即先判断实例是否存在,不存在的情况,再实例化。但是只能在单线程的情况下使用。在多线程的情况下,假如A线程进入到if(instance == null), 但是B线程已经执行到 instance = new Singleton(), 这种情况下,就会产生产生多个实例,未达到单例的效果。

二. 懒汉(线程安全,同步方法)

public class Singleton {

    private static Singleton instance;

    private Singleton() {

    }

    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

总结:通过在方法前加关键字 synchronized来起到线程同步的作用,解决线程不安全。具备lazy loading效果。

缺点:效率太低。每个线程在执行getInstance() 方法时,都会进行同步。实际上,该方法仅仅需要在第一次执行时需要同步,99%情况下不需要同步,所以影响性能。

三. 懒汉(同步代码块)

public class Singleton {
    private static Singleton instance;

    private Singleton() {

    }

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                instance = new Singleton();
            }
        }
        return instance;
    }
}

总结: 同步了代码块,但是仔细分析,多线程下,也会出现 写法一的情况,有可能会产生多个实例。

四. 饿汉(静态变量)

public class Singleton {
    private final static Singleton instance = new Singleton();

    private Singleton() {

    }

    public static Singleton getInstance() {
        return instance;
    }
}

优点:写法简单,在类装载阶段就完成实例化(单例模式中,大多数是调用getInstance()方法时类装载)。基于classloader机制,避免线程同步问题。

缺点:没有lazy loading效果。如果一直没有使用这个实例,会造成内存的浪费。

五. 饿汉(静态代码块)

public class Singleton {
    private static Singleton instance;

    static {
        instance = new Singleton();
    }

    public static Singleton getInstance() {
        return instance;
    }
}

总结: 和上面的,也就是第四种效果差不多。只是把实例化过程放在了静态代码块中,也是在类装载的时候,执行静态代码块,初始化类的实例。

六. 双重校验锁

public class Singleton {
    private static volatile Singleton instance;

    private Singleton() {

    }

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

总结:因为进行了两次if(instance == null)的检查,保证了线程的安全性。

优点:具有lazy loading效果;线程安全;效率高。

缺点:使用了volatile 关键字,JDK1.5以后才支持。

七. 静态内部类

public class Singleton {

    private static class SingletonKeeper {
        private static final Singleton instance = new Singleton();
    }

    private Singleton() {

    }

    public static Singleton getInstance() {
        return SingletonKeeper.instance;
    }
}

总结: 利用classloader的机制来保证初始化instance实例时只有一个线程。和写法四 和 写法五,还是有区别的(只要Singleton类被装载,那么instance就会被实例化,也就没有lazy loading效果)。 而此写法,当Singleton类被装载了,instance不一定被实例化,因为SingletonKeeper类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonKeeper类,从而实例化instance。

类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。

优点:避免了线程不安全,延迟加载,效率高。

八. 枚举

public enum Singleton {
    INSTANCE;
    public void whateverMethod() {

    }
}

总结: 借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值