一、单例模式的产生背景
现在来做这么一件事情,在电脑上打开一个任务资源管理器,然后再次打开,会发现第二次打开时并没有产生新的窗口,还是第一次打开的那个。这种情况非常常见,因为有时确保系统中某个对象的唯一性即一个类只能有一个实例非常重要,比如说Windows 是多进程多线程的,在操作一个文件的时候,就不可避免地出现多个进程或线程同时操作一个文件的现象,所以所有文件的处理必须通过唯一的实例来进行。总的来说,单例模式之所以产生,是因为当实例存在多个会引起程序逻辑错误,需要保持只存在一个类的实例。
二、认识单例模式
单例模式有以下特点:
1、单例类只能有一个实例。
2、单例类必须自己创建自己的唯一实例。
3、单例类必须给所有其他对象提供这一实例。
根据以上特点可知,这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。
三、单例模式的优缺点及使用场景
优点: 1、在内存中只有一个实例,减少了内存开销。 2、避免程序逻辑错误。
缺点:没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。
使用场景:1、要求生产唯一序列号。 2、WEB 中的计数器,不用每次刷新都在数据库里加一次,用单例先缓存起来。 3、创建的一个对象需要消耗的资源过多,比如 I/O 与数据库的连接等。
四、在java中的具体实现
第一种形式:懒汉式,也是常用的形式。
public class SingletonClass{
private static SingletonClass instance=null; //创建 SingleClass 的一个对象
public static synchronized SingletonClass getInstance(){//getInstance() 方法中需要使用同步锁 synchronized (Singleton.class) 防止多线程同时进入造成 instance 被多次实例化。
if(instance==null){
instance=new SingletonClass();
}
return instance;
}
private SingletonClass(){ //让构造函数为 private,这样该类就不会被实例化
}
}
第二种形式:饿汉式
//对第一行static的一些解释
// java允许我们在一个类里面定义静态类。比如内部类(nested class)。
//把nested class封闭起来的类叫外部类。
//在java中,我们不能用static修饰顶级类(top level class)。
//只有内部类可以为static。
public class Singleton{
//在自己内部定义自己的一个实例,只供内部调用
private static final Singleton instance = new Singleton();
private Singleton(){
//do something
}
//这里提供了一个供外部访问本class的静态方法,可以直接访问
public static Singleton getInstance(){
return instance;
}
}
第三种形式: 双重锁的形式。
public class Singleton{
private static volatile Singleton instance=null;
private Singleton(){
//do something
}
public static Singleton getInstance(){
if(instance==null){
synchronized(SingletonClass.class){
if(instance==null){
instance=new Singleton();
}
}
}
return instance;
}
}
//这个模式将同步内容下方到if内部,提高了执行的效率,不必每次获取对象时都进行同步,只有第一次才同步,创建了以后就没必要了。
//这种模式中双重判断加同步的方式,比第一个例子中的效率大大提升,因为如果单层if判断,在服务器允许的情况下,
//假设有一百个线程,耗费的时间为100*(同步判断时间+if判断时间),而如果双重if判断,100的线程可以同时if判断,理论消耗的时间只有一个if判断的时间。
//所以如果面对高并发的情况,而且采用的是懒汉模式,最好的选择就是双重判断加同步的方式。
五、懒汉模式和饿汉模式的区别
1、懒汉式:在使用时才进行对象的实例化,懒汉式的特点是延迟加载,
2、饿汉式:在类加载时进行对象的实例化,饿汉式的特点是一开始就加载。