Java设计模式---单列模式

一、饿汉式

//饿汉式单列
public class Hungry {

    //构造私有化是单列的第一步
    private Hungry(){
    }
    
    private final static Hungry HUNGRY=new Hungry();
    
    public static Hungry getInstance(){
        return HUNGRY;
    }
    
}

这种模式上来就加载 ,如果我们在加几个属性,会发现data1这些数组及其的耗费内存资源。

//饿汉式单列
public class Hungry {
  
    private byte[] data1=new byte[1024*1024];
    private byte[] data2=new byte[1024*1024];
    private byte[] data3=new byte[1024*1024];
    private byte[] data4=new byte[1024*1024];
    //构造私有化是单列的第一步
    private Hungry(){

    }
    private final static Hungry HUNGRY=new Hungry();
    public static Hungry getInstance(){
        return HUNGRY;
    }
}

所以我们可以使用懒汉式

二、懒汉式

当真正使用的时候才去加载

public class LayzMan {

    private LayzMan(){ } //构造器私有

    private static LayzMan layzMan;

    public static LayzMan getInstance(){
        if(layzMan==null){
            layzMan=new LayzMan();
        }
        return layzMan;
    }
}

但是这种单列只有在单线程下是ok的,如果在多线程下:


public class LayzMan {

    private LayzMan(){
        System.out.println(Thread.currentThread().getName()+"ok");
    } //构造器私有

    private static LayzMan layzMan;

    public static LayzMan getInstance(){
        if(layzMan==null){
            layzMan=new LayzMan();
        }
        return layzMan;
    }
		
    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                LayzMan.getInstance();
            }).start();
        }
    }
}
//第一次打印结果
Thread-1ok
Thread-2ok
Thread-0ok

//第二次
Thread-0ok

会发现如果是多线程的情况下每次都不一样。有时会new多次
所以我们为了保证多线程情况下单列也能被使用,就需要加锁

public class LayzMan {

    private LayzMan(){
        System.out.println(Thread.currentThread().getName()+"ok");
    } //构造器私有

    private static LayzMan layzMan;

    //双重检测锁模式,DCL懒汉式单例
    public static LayzMan getInstance(){
        if(layzMan==null){
            synchronized (LayzMan.class){
                if(layzMan==null){
                    layzMan=new LayzMan();
                }
            }
        }
        return layzMan;
    }

    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                LayzMan.getInstance();
            }).start();
        }
    }
}

//打印多次都会只会出现
Thread-0ok 

这种方式简称为DCL懒汉式,但是这种方式也不安全

    public static LayzMan getInstance(){
        if(layzMan==null){
            synchronized (LayzMan.class){
                if(layzMan==null){
                    layzMan=new LayzMan();
                    /*
                    *   new LayzMan();不是原子性操作
                    *   底层操作步骤分别是
                    *   1.分配内存空间
                    *   2.执行构造方法,初始化对象
                    *   3.把这个对象指向这个空间
                    *  正常情况下是 1 2 3按照步骤执行
                    *    但也可能出现 132 的情况 就是先走3
                    *    如果A线程走的是132 当A还没执行2时 
                    *    ,B线程就 以为A构造完了直接走了下面的return
                    *
                    * */
                }
            }
        }
        return layzMan;
    }

所以 我们一定得保证构建的顺序性,需要在被构建的对象上加 volatile关键字

  private volatile static LayzMan layzMan;

完整的DCL懒汉式


public class LayzMan {

    private LayzMan(){
        System.out.println(Thread.currentThread().getName()+"ok");
    } //构造器私有

    private volatile static LayzMan layzMan; //一定记得加上volatile 关键字

    public static LayzMan getInstance(){
        if(layzMan==null){
            synchronized (LayzMan.class){
                if(layzMan==null){
                    layzMan=new LayzMan();
                }
            }
        }
        return layzMan;
    }

    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                LayzMan.getInstance();
            }).start();
        }
    }
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值