JavaDLC单例模式

Java DLC单例模式

懒汉式单例模式

public void Singleton{
    //私有化静态对象
    private static Singleton singleton;
    //私有化构造方法
    private Singleton(){}
    //公有的get对象方法
    public static Singleton getInstance(){
        if(singleton==null){
            //当需要调用是,才会new对象
            singleton=new Singleton();
        }
        return singleton;
    }
}

这种方法在单线程模式下是可取的,但是在并发情况下无法保证线程安全,可以加一个synchronized关键字进行优化

public void Singleton{
    //私有化静态对象
    private static Singleton singleton;
    //私有化构造方法
    private Singleton(){}
    //公有的get对象方法
    public static Singleton getInstance(){
        synchronized(Singleton.class) {
            if(singleton==null){
                //当需要调用是,才会new对象
                singleton=new Singleton();
            }
        }
        return singleton;
    }
}

但是上述代码也有问题,singleton=new Singleton();分为三步执行,首先会为对象分配内存空间,然后初始化对象,最后返回对象的引用,但由于jvm编译器的优化产生的重排序缘故,步骤2、3可能会发生重排序,可能会导致return的对象为null

饿汉式单例模式

public void Singleton{
    //直接new好对象
    private static Singleton singleton = new Singleton();

    private Singleton(){}

    public static Singleton getInstance(){
        return singleton;
    }
}

DCL:Double Checked Locking (双重检查锁定)

​单例模式(懒汉式的基础上加入并发控制):在多线程环境下,对外存在一个对象

  • 1.构造器私有化(避免外部new构造器)

  • 2.提供私有的静态属性(存储对象的地址)

  • 3.提供共有的静态方法(获取属性)

    public class TestDcl{
        //提供私有的静态属性(存储对象的地址)
        //加入volatile防止其他线程可能访问一个为null的对象
        private volatile static TestDcl instance;
    
        //构造器私有化(避免外部new构造器)
        private TestDcl() {
    
        }
    
        public static TestDcl getInstance() {
            //double检测
            //如果已经有对象,防止再次创建
            if (null != instance) {
            	return instance;
            }
            //有两个线程A B,A进来发现无对象,创建对象
            //此时B也进来,发现无对象,也创建对象,这样就有两个对象
            synchronized(TestDcl.class) {
                if (instance == null) {
                    //1.为对象分配内存空间    2.初始化对象    3.返回对象的引用
                    //但由于jvm编译器的优化产生的重排序缘故,步骤2、3可能会发生重排序
                    //利用volatile的特性即可阻止重排序和可见性
                    instance = new TestDcl();
                }
            }
            return instance;
        }
    
        public static void main(String[] args) {
            new Thread(()->{
                System.out.println(TestDcl.getInstance().toString());
            })start();
    
            System.out.println(TestDcl.getInstance());
    
        }
    }
    
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值