系统中有些资源只能有一个,或者一个就够,多个浪费。例如一个系统只能有一个窗口管理器或文件系统、一个系统只能有一个计时器或序号生成器、web系统只能有一个页面计数器等等。此时,最好就需要把这些资源设置成有且仅有一个实例。 代码中也就是如何保证一个类只有一个实例并且这个实例能够被访问呢?只有一个实例的就意味着不能让其他类来实例化,也就是只能自己实例化自己。能够被访问也就意味着自身要对外提供全局方法来获取到这个实例,这就是单例模式。 单例模式定义:确保某一个类只有一个实例,而且自行实例化并且向整个系统提供这个实例。 单例模式通常代表着系统具有唯一性的资源。主要有3点:只有一个实例;自行创建这个实例;自行向整个系统提供这个实例。
二、结构与类图
单例模式是创建型模式,其实结构非常简单,需要注意以下3点: 1、构造方法私有:不让外部实例化,只能将构造函数私有; 2、提供一个公共静态方法获取实例:获取这个实例前是没有实例的,只能用静态的。 3、实例保存到自身静态私有属性上:获取方法是静态的,实例当然也只能是静态的,最好是final的,单例不允许修改; 通用类图如下:
代码如下:
public class Singleton {
private static final Singleton instance = new Singleton();
private Singleton() {
}
public static Singleton getInstance() {
return instance ;
}
}
三、适用场景及效果(优缺点)
四、示例
比如要做一个页面计数器,可以使用单例模式,非常简单,直接看代码
//页面计数器
public class PageCounter {
private static final PageCounter instance = new PageCounter();
// 计数器
private AtomicLong counter = new AtomicLong(0);
private PageCounter() {
}
public static PageCounter getInstance() {
return instance ;
}
public void add() {
counter.getAndAdd(1);
}
public long get() {
return counter.get();
}
}
五、模式扩展
说到单例模式,很多人想到的是如何创建单例模式,有很多种创建方法,懒汉、恶汉、双重锁等等,此处大概介绍一下。 第一种(饿汉)
//饿汉模式(推荐),类加载时就创建了
//优点:1、线程安全;2、调用getInstance时速度快
//缺点:1、无法延迟加载;2、有可能浪费资源,无人调用getInstance()时,仍然创建了实例
public class Singleton01 {
private static final Singleton01 instance = new Singleton01();
private Singleton01() {
}
public static Singleton01 getInstance() {
return instance ;
}
}第二种(饿汉变种)
//饿汉模式变种,类加载时就创建了,和上一个模式区别不大,只是能在static中加入逻辑处理
public class Singleton02 {
private static Singleton02 instance = null;
static {
// 此处可以写一些逻辑
instance = new Singleton02();
}
private Singleton02() {
}
public static Singleton02 getInstance() {
return instance ;
}
}第三种(懒汉)
//懒汉(线程不安全),用到了再去初始化
//优点:延迟加载
//缺点:致命的并发问题,可能导致创建多次
public class Singleton03 {
private static Singleton03 instance = null;
private Singleton03() {
}
public static Singleton03 getInstance() {
if ( instance == null ) {
// 此处有并发问题
instance = new Singleton03();
}
return instance ;
}
}第四种(懒汉变种)
//懒汉(线程安全)
//优点:延迟加载
//缺点:效率低下,初始化完毕后,getInstance()方法根本不需要同步了
public class Singleton04 {
private static Singleton04 instance = null;
private Singleton04() {
}
public synchronized static Singleton04 getInstance() {
if ( instance == null ) {
instance = new Singleton04();
}
return instance ;
}
}第五种(双重锁定检查)
//双重锁定检查
//优点:延迟加载,效率高
//缺点:jdk1.5之后才可以用
//由于jdk1.5之前编译器允许处理器乱序执行,所以可能导致获取到没初始化完毕的instance
public class Singleton05 {
private static Singleton05 instance = null