概念
单例模式Singleton也是一种用于创建对象的设计模式
单例模式用于一个单一的类,让这个类自己负责创建自己的实例对象,并对外提供一个访问该实例对象的方法
单例模式有多种类型,包括饿汉式、懒汉式(分线程安全和线程不安全)、双重验证、静态内部类、枚举单例
特点
单例类只能有一个实例,这个实例只能由类自身创建
单例类必须对外提供一个能够获取这个实例的方法
懒汉式是在调用获取实例的方法时创建实例
饿汉式实在编译时创建实例,调用获取实例的方法时可以立即返回已创建好的实例
使用场景
某个类仅用于操作其他对象或文件,并且这个类的实例会频繁的被创建和销毁,比如对数据库的读写操作辅助类,需要读写数据库时实例一个对象,读写完成后销毁这个对象。当数据库的操作很频繁时,就会频繁地打开和关闭数据库,容易出现“锁表”,并且容易发生数据读写不同步的问题。如果使用单例模式,所有对数据库的操作都基于同一个操作对象,只需要打开一次数据库,对数据库的读写操作将会依次进行,即提高了效率又减少了消耗。
懒汉式和饿汉式之间推荐使用饿汉式,饿汉式的执行效率较高
需要懒加载时应该用静态内部类实现单例模式
代码示例
一个最简单的懒汉式(线程不安全)
public class SingletonObject {
//唯一的实例对象
private static SingletonObject sInstance;
//外部获取实例对象的方法
public static SingletonObject getInstance(){
if (sInstance == null){
sInstance = new SingletonObject();
}
return sInstance;
}
//构造方法私有,不允许外部创建实例对象
private SingletonObject(){
System.out.println("实例对象被创建");
}
}
一个最简单的饿汉式
public class SingletonObject {
//唯一的实例对象
private static SingletonObject sInstance = new SingletonObject();
//外部获取实例对象的方法
public static SingletonObject getInstance(){
return sInstance;
}
//构造方法私有,不允许外部创建实例对象
private SingletonObject(){
System.out.println("实例对象被创建");
}
}
线程安全的懒汉式
//外部获取实例对象的方法
//使用synchronized修饰符,防止不同线程同时调用这个方法
public static synchronized SingletonObject getInstance(){
if (sInstance == null){
sInstance = new SingletonObject();
}
return sInstance;
}
双重验证,在线程安全的懒汉式中,如果在某次调用执行到if(sInstance == null)时立即执行另一次调用(第一次调用还没用执行到对象创建的语句,此时sInstance依然为null),就会发生两次执行的if判断都为true,从而创建两次对象实例
//外部获取实例对象的方法
//使用synchronized修饰符,防止不同线程同时调用这个方法
public static synchronized SingletonObject getInstance(){
if (sInstance == null){
//在方法内部再次同步,防止出现不同线程先后进入这个方法后,都创建了对象实例的情况
synchronized(SingletonObject.class){
if (sInstance == null){
sInstance = new SingletonObject();
}
}
}
return sInstance;
}
静态内部类,在内部类中以常量的形式来保存实例对象,使得对象唯一且不可变,实现了线程安全
//保存对象实例的静态内部类
private static class SingletonHolder{
private static final SingletonObject INSTANCE = new SingletonObject();
}
public SingletonObject getInstance(){
return SingletonHolder.INSTANCE;
}
以上就是单例模式的几种常用的实现方式,枚举实现在实际的开发中基本不会使用,在此不再举例介绍