java 单例 同步_Java单例和同步

问题

请澄清我对Singleton和Multithreading的疑问:

在多线程环境中,在Java中实现Singleton的最佳方法是什么?

当多个线程同时尝试访问getInstance()方法时会发生什么?

我们可以使单例的getInstance()同步吗?

使用Singleton类时是否真的需要同步?

#1 热门回答(181 赞)

是的,这是必要的。使用延迟初始化可以使用多种方法来实现线程安全:

严厉同步:

private static YourObject instance;

public static synchronized YourObject getInstance() {

if (instance == null) {

instance = new YourObject();

}

return instance;

}

这个解决方案要求将每个线程同步,而实际上只需要前几个线程。

private static final Object lock = new Object();

private static volatile YourObject instance;

public static YourObject getInstance() {

YourObject r = instance;

if (r == null) {

synchronized (lock) { // While we were waiting for the lock, another

r = instance; // thread may have instantiated the object.

if (r == null) {

r = new YourObject();

instance = r;

}

}

}

return r;

}

此解决方案确保只有尝试获取单例的前几个线程必须经历获取锁定的过程。

private static class InstanceHolder {

private static final YourObject instance = new YourObject();

}

public static YourObject getInstance() {

return InstanceHolder.instance;

}

此解决方案利用Java内存模型对类初始化的保证来确保线程安全。每个类只能加载一次,只有在需要时才会加载。这意味着第一次调用getInstance,InstanceHolder将被加载并且将创建instance,并且由于这是由ClassLoaders控制的,因此不需要额外的同步。

#2 热门回答(56 赞)

这种模式通过显式同步对instance进行线程安全的延迟初始化!

public class MySingleton {

private static class Loader {

static final MySingleton INSTANCE = new MySingleton();

}

private MySingleton () {}

public static MySingleton getInstance() {

return Loader.INSTANCE;

}

}

它的工作原理是因为它使用类加载器为你自由地进行所有同步:classMySingleton.Loader首先在getInstance()方法中访问,所以Loaderclass在第一次调用getInstance()时加载。此外,类加载器保证在你访问类之前完成所有静态初始化 - 这就是为你提供线程安全性的方法。

这就像魔术一样。

它实际上非常类似于Jhurtado的枚举模式,但是我发现枚举模式是对枚举概念的滥用(虽然它确实有效)

#3 热门回答(22 赞)

如果你正在使用Java中的多线程环境并且需要保证所有这些线程都在访问类的单个实例,则可以使用Enum。这将有助于你处理序列化的附加优势。

public enum Singleton {

SINGLE;

public void myMethod(){

}

}

然后让你的线程使用你的实例,如:

Singleton.SINGLE.myMethod();

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值