Singleton Pattern 两种模式

本文详细介绍了两种Java中实现单例模式的方法:饿汉式和双重校验的懒汉式。饿汉式在类加载时即创建单例,保证线程安全。双重校验懒汉式则在多线程环境下通过 volatile 和双重校验避免同步带来的性能开销,确保单例的正确创建。同时解释了 volatile 关键字如何防止指令重排序,避免多线程问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

public class Singleton {
	private Singleton(){}
	private static final Singleton instance = new Singleton();
	public static Singleton getInstance(){
		return instance;
	}
}

这是一种饿汉且保证线程安全的模式(强烈推荐)。

private static final Singleton instance = new Singleton();为什么要用static,因为getInstance方法是static的(不用生成实例就能使用),静态方法只能调用静态成员,在类初始化时就实例化instance。

public class Singleton {//双重校验的懒汉,且线程安全
	private Singleton(){}
	private volatile static Singleton instance;(加volatile防止指令重排序)
	public static Singleton getInstance(){
		if(instance == null){//加锁效率低,在已经生成实例后,没必要再判断锁
			synchronized(Singleton.class){//加锁,防止多线程时,生成多个实例
				if(instance == null){
					instance = new Singleton();指令重排序,先完成赋值,但构造函数还没执行完
				}
			}
		}
		return instance;
	}	
}

instance = new Singleton() 可以分解为3行伪代码:
1 memory=allocate();// 分配内存 相当于c的malloc
2 ctorInstanc(memory) //初始化对象
3 instance=memory //设置instance指向刚分配的地址
上面的代码在编译器运行时,可能会出现重排序 从1-2-3 排序为1-3-2
如此在多线程下就会出现问题。
例如现在有2个线程A,B,线程A在执行第5行代码时,B线程进来,而此时A执行了 1和3,没有执行2,此时B线程判断instance不为null 直接返回一个未初始化的对象,就会出现问题,而用了volatile,上面的重排序就会在多线程环境中禁止,不会出现上述问题。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值