Java 单例模式的安全实现方式

1. 通过双重检查加锁机制实现线程安全的单例模式

/**
 * 
 * 双重检查加锁机制:实现线程安全的单例模式
 */
public class Singleton {
	/**
	 * 双重检查加锁机制的实现一般会使用一个关键字volatile,它的意思是:被volatile
	 * 修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。
	 * 
	 * 提示:由于volatile关键字可能会屏蔽掉虚拟机中的一些必要的代码优化,所以运行效率并不是很高,
	 * 因此一般建议,没有特别的需要,不要使用。也就是说,虽然可以使用“双重检查加锁”机制实现线程安全的实例, 但并不建大量采用,可以根据情况来采用
	 */
	private volatile static Singleton instance = null;

	private Singleton() {
	}

	public static Singleton getInstance() {
		// 先检查实例是否存在,如果不存在才进入下面的同步块
		if (instance == null) {
			// 同步块,线程安全地创建实例
			synchronized (Singleton.class) {
				// 再次检查实例是否存在,如果不存在才真正地创建实例
				instance = new Singleton();
			}
		}
		return instance;
	}
}


2. 一种更好的安全单例实现方式

/**
 * 一种更好的安全单例实现方式:既能实现延迟加载,又能实现线程安全
 * 
 * 这个实现方式综合使用了Java的类级内部类和多线程缺省同步锁的知识,很巧妙地同时实现了延迟加载和线程安全
 */
public class Singleton2 {

	/**
	 * 类级的内部类,也就是静态的成员式内部类,该内部类的实例与外部类的实例没有绑定关系,而且只有被调用到时才会装载,从而实现了延迟加载
	 * 
	 */
	private static class SingletonHolder {
		/**
		 * 静态初始化器,由JVM来保证线程安全
		 */
		private static Singleton2 instance = new Singleton2();
	}

	/**
	 * 私有化构造方法
	 */
	private Singleton2() {
	}

	/**
	 * 这个模式的优势在于:getInstance方法并没有被同步,并且只是执行一个域的访问,因此延迟初始化并没有增加任何访问成本
	 * 
	 */
	public static Singleton2 getInstance() {
		return SingletonHolder.instance;
	}
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值