1 概念
单例即一个类仅拥有一个实例对象。
单例的实现主要是通过以下两个步骤:
- 将该类的构造方法定义为私有方法,这样其他处的代码就无法通过调用该类的构造方法来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例;
- 在该类内提供一个静态方法,当我们调用这个方法时,如果类持有的引用不为空就返回这个引用,如果类保持的引用为空就创建该类的实例并将实例的引用赋予该类保持的引用
2 饿汉式单例实现
public class Signleton{
private Singleton(){
}
private static Signleton s = new Signleton();
pulic getInstance(){
return s;
}
}
优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。
缺点:在类装载的时候就完成实例化,没有达到Lazy Loading的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费。
3 懒汉式
3.1 实现1-线程不安全
public class Signleton {
private Signleton(){}
private static Signleton signleton;
public static Signleton getInstance(){
if (signleton==null)
return new Signleton();
return signleton;
}
}
线程不安全,当两个线程同时在if判断时,会产生两个实例。
3.2 懒汉式-方法加锁
public class Signleton {
private Signleton(){}
private static Signleton signleton;
public static synchronized Signleton getInstance(){
if (signleton==null)
return new Signleton();
return signleton;
}
}
缺点:整个方法加锁,效率太低。只有构造实例需要加锁。
3.3 懒汉式-代码块加锁
public static Signleton getInstance1(){
if (signleton==null)
synchronized(Signleton.class){
return new Signleton();
}
return signleton;
}
缺点:线程不安全,假如一个线程进入了if (singleton == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例。
3.3 懒汉式-双重判断
public static Signleton getInstance1(){
if (signleton==null)
synchronized(Signleton.class){
if (signleton==null)
return new Signleton();
}
return signleton;
}
优点:线程安全;延迟加载;效率较高。
4 内部静态类
private static class SignletonInstance{
private static final Signleton signleton = new Signleton();
}
public static Signleton getInstance(){
return SignletonInstance.signleton;
}
这种方式跟饿汉式方式采用的机制类似,但又有不同。两者都是采用了类装载的机制来保证初始化实例时只有一个线程。不同的地方在饿汉式方式是只要Singleton类被装载就会实例化,没有Lazy-Loading的作用,而静态内部类方式在Singleton类被装载时并不会立即实例化,而是在需要实例化时,调用getInstance方法,才会装载SingletonInstance类,从而完成Singleton的实例化。
类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。
优点:避免了线程不安全,延迟加载,效率高。
5 枚举实现
public enum EnumSignelton {
INSTANCE;
private Resource instance;
EnumSignelton(){
instance = new Resource();
}
public Resource getInstance() {
return instance;
}
}
EnumSignelton.INSTANCE.getInstance() 即可获得所要实例。
首先,在枚举中我们明确了构造方法限制为私有,在我们访问枚举实例时会执行构造方法,同时每个枚举实例都是static final类型的,也就表明只能被实例化一次。在调用构造方法时,我们的单例被实例化。 也就是说,因为enum中的实例被保证只会被实例化一次,所以我们的INSTANCE也被保证实例化一次。
枚举也提供了序列化机制。某些情况,比如我们要通过网络传输一个数据库连接的句柄,会提供很多帮助。
6 应用
优点
系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能。
缺点
当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。
适用场合
- 需要频繁的进行创建和销毁的对象;
- 创建对象时耗时过多或耗费资源过多,但又经常用到的对象;
- 工具类对象;
- 频繁访问数据库或文件的对象。