2 zksfxy zksfxy 于 2016.02.18 11:19 提问

关于单利模式中的懒人模式

最近面试,被要求写多线程并发状态下的单例模式,随写了懒人模式

 public class Singleton
{
    private Singleton(){}
    public static readonly Singleton instance = new Singleton();
}  

被问道,如何在多线程并发的状态下保证单例模式,查了半天对readonly 是如何保证并发状态的下,多个线程同时调用instance变量而产生单例的原理没有找到,求大神给讲讲呗!

7个回答

qq_16414307
qq_16414307   2016.02.21 16:28
已采纳

你的代码,的确可以保证单例, 静态属性初始化,系统会保证线程安全的

但是你的代码不是延迟模式,或懒人模式噢

只要任何使用到这个类,无论什么方法,就会初始化静态变量,
比如这个类如果还有其他静态方法,那么访问这些方法时,就会建立对象了

延迟创建,肯定要求只访问这个属性时才建立对象,所以你必须写一个静态方法,在方法里创建对象

这里考虑效率,无非就是检查两次对象是否建立
if (o == null){
锁定
if (o==null) o=new ...
释放
}
return o

对比代码
if (o==null) o=new
如果两个线程同时检查o,当然都是null,然后两个同时创建对象,等于就创建两次了

而代码
锁定
if(o==null) o=new
释放

则每次执行都要锁,效率太低了,

qq_23660243
qq_23660243   2016.02.18 11:39
bdmh
bdmh   Ds   Rxr 2016.02.18 11:30

你这个根本就不是单例模式,原理应该是,内部静态实例,保证创建一次

zksfxy
zksfxy 如果不用锁的方式,只用readonly关键字,能实现不?
接近 2 年之前 回复
wangyaninglm
wangyaninglm   Ds   Rxr 2016.02.18 11:35

需要用锁来保证其安全性,使用double-check 来保证thread safety

if(null == instance)
{
lock();
if(null == instance)
{
instance = new Singleton();
}
unlock();

return instance;
}

qq_18748427
qq_18748427   2016.02.18 11:41

加锁。synchronized。
public class Singleton
{
private Singleton(){}
private staitc Singleton singleton = null;
public static synchronized Singleton getSingleton(){
if(singleton = null) singleton = new Singleton();
return singleton;
}
}

Mr_dsw
Mr_dsw   Ds   Rxr 2016.02.18 22:51

保证线程安全就是需要同步,单利模式有懒汉和恶汉两个模式

qq_20039385
qq_20039385   2016.02.19 11:11

用下面这种懒汉式即线程安全效力又高。
图片说明

Csdn user default icon
上传中...
上传图片
插入图片
准确详细的回答,更有利于被提问者采纳,从而获得C币。复制、灌水、广告等回答会被删除,是时候展现真正的技术了!