定义:
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
一、确保一个类只有一个实例被建立
二、提供了一个对对象的全局访问指针
三、在不影响单例类的客户端的情况下允许将来有多个实例
经典的单例模式有三种,懒汉式、饿汉式和 登记式。
- 饿汉式(一开始就加载了):在程序启动或单件模式类被加载的时候,单件模式实例就已经被创建。
//饿汉式单例模式
public class MySingleton2 {
//设立静态变量,直接创建实例
private static MySingleton2 mySingleton = new MySingleton2();
private MySingleton2(){
//私有化构造函数
System.out.println("-->饿汉式单例模式开始调用构造函数");
}
//开放一个公有方法,判断是否已经存在实例,有返回,没有新建一个在返回
public static MySingleton2 getInstance(){
System.out.println("-->饿汉式单例模式开始调用公有方法返回实例");
return mySingleton;
}
}
- 懒汉式(延迟加载):当程序第一次访问单件模式实例时才进行创建。
//懒汉式单例模式
public class MySingleton {
//设立静态变量
private static MySingleton mySingleton = null;
private MySingleton(){
//私有化构造函数
System.out.println("-->懒汉式单例模式开始调用构造函数");
}
//开放一个公有方法,判断是否已经存在实例,有返回,没有新建一个在返回
public static MySingleton getInstance(){
System.out.println("-->懒汉式单例模式开始调用公有方法返回实例");
if(mySingleton == null){
System.out.println("-->懒汉式构造函数的实例当前并没有被创建");
mySingleton = new MySingleton();
}else{
System.out.println("-->懒汉式构造函数的实例已经被创建");
}
System.out.println("-->方法调用结束,返回单例");
return mySingleton;
}
}
优缺点
懒汉式是典型的时间换空间
饿汉式是典型的空间换时间不加同步的懒汉式是线程不安全的。比如,有两个线程,一个是线程A,一个是线程B,它们同时调用getInstance方法,就可能导致并发问题。
加上synchronized即可实现懒汉式的线程安全
public static synchronized Singleton getInstance(){}饿汉式是线程安全的,因为虚拟机保证只会装载一次,在装载类的时候是不会发生并发的。
/**
* 双重检查加锁的单例模式
* @author dream
*
*/
public class Singleton {
/**
* 对保存实例的变量添加volitile的修饰
*/
private volatile static Singleton instance = null;
private Singleton(){
}
public static Singleton getInstance(){
//先检查实例是否存在,如果不存在才进入下面的同步块
if(instance == null){
//同步块,线程安全的创建实例
synchronized (Singleton.class) {
//再次检查实例是否存在,如果不存在才真正的创建实例
instance = new Singleton();
}
}
return instance;
}
}
public class Singleton {
/**
* 类级的内部类,也就是静态类的成员式内部类,该内部类的实例与外部类的实例
* 没有绑定关系,而且只有被调用时才会装载,从而实现了延迟加载
* @author dream
*
*/
private static class SingletonHolder{
/**
* 静态初始化器,由JVM来保证线程安全
*/
private static final Singleton instance = new Singleton();
}
/**
* 私有化构造方法
*/
private Singleton(){
}
public static Singleton getInstance(){
return SingletonHolder.instance;
}
}
单元素的枚举类型已经成为实现Singleton的最佳方法。
/**
* 使用枚举来实现单例模式的示例
* @author dream
*
*/
public class Singleton {
/**
* 定义一个枚举的元素,它就代表了Singleton的一个实例
*/
uniqueInstance;
/**
* 示意方法,单例可以有自己的操作
*/
public void singletonOperation(){
//功能树立
}
}