16.单例模式与多线程

16.单例模式与多线程

所谓单例,最重要的一个思想就是构造器私有,这样别人就没办法去new这个对象了。

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。
注意:
1、单例类只能有一个实例。
2、单例类必须自己创建自己的唯一实例。
3、单例类必须给所有其他对象提供这一实例。

主要解决:一个全局使用的类频繁地创建与销毁。

单例的实现思路

  • 静态化实例对象
  • 私有化构造方法,禁止通过构造方法创建实例
  • 提供一个公共的静态方法,用来返回唯一实例

单例的好处

  • 只有一个对象,内存开支少、性能好
  • 避免对资源的多重占用
  • 在系统设置全局访问点,优化和共享资源访问

16.1饿汉式单例(不推荐)

饿汉式单例一上来就new对象会造成资源的浪费,而且多线程会破坏单例原则。

package com.single;

/**
 * Created by yj on 2020/8/30 20:33
 */
//1.饿汉式单例,饿汉式单例一上来就new对象会造成资源的浪费
public class Hungry01 {
    //构造器私有
    private Hungry01(){
    }
    //new一个对象
    private final static Hungry01 HUNGRY = new Hungry01();
    //创建对象返回方法
    public static Hungry01 getInstance(){
        return HUNGRY;
    }

}

16.2懒汉式单例(多线程不推荐)

懒汉式单例,它是在要使用对象的时候才加载。

  • 饿汉式单例用一个if去判断当前实例对象是否生成,未生成的话就new一个,已经生成了就直接返回这个对象实例
  • 在多线程下是不安全的,要求的是任何线程其拿到的对象实例应该是一个对象实例,因为单例就是只返回一个对象实例,也就是只有第一个线程能用以下构造器,其他的都不能使用构造器,直接拿到对象实例返回即可。
package com.single;

import java.util.TreeMap;

/**
 * Created by yj on 2020/8/30 20:47
 */
//2.懒汉式单例,它是在要使用对象的时候才加载
public class LazyMan02 {
    private LazyMan02(){
        System.out.println(Thread.currentThread().getName()+"====>ok");
    }
    private static LazyMan02 lazyMan;
    //用一个if去判断当前实例对象是否生成,未生成的话就new一个,已经生成了就直接返回这个对象实例
    //但是这个在多线程下是不安全的,要求的是任何线程其拿到的对象实例应该是一个对象实例,因为单例就是只返回一个对象实例
    public static LazyMan02 getInstance(){
        if(lazyMan==null){
            lazyMan = new LazyMan02();
        }
        return lazyMan;
    }
    //单线程下单例,没问题,多线程单例出现问题
    public static void main(String[] args) throws Exception {
        for(int i=0;i<10;i++){
            new Thread(()->{
                LazyMan02.getInstance();
            }).start();
        }
    }

}

16.3dcl双重校验锁懒汉式(多线程推荐,存在隐患)

可以看到这样构造器就初始化了一次,但是有10个线程操作了这个单例。

  • 第一个if是判断是否当前这个实例对象已经创建。
  • 第二个if原因:第一个线程进入第一个if,拿到锁进入第二个if并创建实例对象,此时仍然未创建完对象,而其他线程在锁这儿阻塞,如果没有第二个if,只有第一个if的话其他线程已经进入第一个if后阻塞了,这样就会创建一个新的实例对象,违背单例原则,但是如果有第二个if,此时第一个线程实例对象已经创建了,后面的线程判断出来实例对象不为空了,就会退出去返回第一个线程创建的实例。
  • volatile原因主要是因为它能够禁止指令重排,因为new一个对象不是一个原子性操作,它分为分配内存,初始化对象,对象指向空间,如果第一个线程创建对象实例时候指令重排时候其顺序乱了(先对象指向空间,还未完成对象初始化),那么此时第二个线程在第一个if处判断出实例对象已经有了,它会直接返回实例对象,而此时这个第一个线程只是将对象指向了空间,还未完成对象的构造,就会出错。
package com.single;

/**
 * Created by yj on 2020/8/30 20:47
 */
//3.dcl双重校验锁形式
public class DCLazyMan03 {
    private DCLazyMan03(){
        System.out.println(Thread.currentThread().getName()+"构造器+====>ok");
    }

    private volatile static DCLazyMan03 lazyMan;//加这个原因是为了防止指令重排

    public static DCLazyMan03 getInstance(){
        if(lazyMan==null){
            synchronized (DCLazyMan03.class){
                //为什么这儿要再校验一次是否是null,因为第一个线程在进入第一个if以后拿到锁了,
                //由于此时lazyMan仍然是空的,那么其他线程能进入第一个if,这时候其他线程都在阻塞,当第一个线程创建完实例以后
                //如果没有第二个if,其他线程会进入里面,再创建一个实例,这就违背了单例的原则了,所以再加一个判断,第一个
                //线程在进入这个if以后,在它刚创建实例对象后,释放了锁。其他正在阻塞的线程也进入了第一个if,拿到锁后,发现此时
                //这个已经不为null了,所以此时就不会进入,直接返回已经创建好的实例,维护了单例模式。
                if(lazyMan==null){
                    /**其实下面这步在底层编译为class文件的时候是三个步骤
                    * 1.在堆中分配对象内存
                     *2.执行构造方法,初始化对象
                     *3.把对象指向这个空间
                     *编译器编译代码的时候可能会先指向1,2,3,但是也有可能指向1,3,2
                     *如果此时线程A编译后执行的是1,3,2,由于此时对象已经指向了这个空间,那么线程B进来的时候就判断不为null,返回
                     *直接返回实例,此时就会出现一个问题,就是当前的对象其实没有完成构造,这时候返回就有问题。
                     * 所以上面要加volatile
                     */
                    lazyMan = new DCLazyMan03();//这个并不是一个原子性操作,它会分成三个步骤执行,这时候编译的时候会出现指令重排现象
                }
            }

        }
        return lazyMan;
    }
    public static void main(String[] args) throws Exception {
        for(int i=0;i<10;i++){
            new Thread(()->{
                DCLazyMan03.getInstance();
                System.out.println(Thread.currentThread().getName()+"线程+====>ok");
            }).start();
        }
    }

}
image-20200905101040705

16.4静态内部类(不推荐)

package com.single;

/**
 * Created by yj on 2020/9/4 23:14
 */
//4.静态内部类
public class Holder04 {
    private Holder04(){

    }

    public static Holder04 getInstance(){
        return InnerClass.HOLDER_04;
    }

    public static class InnerClass{
        private static final Holder04 HOLDER_04 = new Holder04();
    }
}

16.5单例模式的安全性问题(解决懒汉式隐患)

单例模式可以被反射破坏原则,它可以通过反射去创建一个新的实例,解决方法就是设置一个加密的变量放在构造器处,默认为false,如果这个构造器被调用过了就为ture,以此来判断这个构造器是否可用,这样就保证了这个构造器只能被用一次,保证了单例的原则.

这样做不论即使是两次使用反射去创建实例,他也能判断出来,抛出异常从而解决问题.

但是这样做,如果被外部猜出了这个secrect标量,那么也是会产生问题的,可以通过反射去修改这个标量的值.

package com.single;

import java.lang.reflect.Constructor;

/**
 * Created by yj on 2020/8/30 20:47
 */
//5.解决反射破坏单例情况
public class ReflectDCLazyMan05 {

    private static boolean secrect = false;
    private ReflectDCLazyMan05(){
        synchronized (ReflectDCLazyMan05.class){
            if(secrect==false){
                secrect = true;
            }
            else{
                throw new RuntimeException("不要用反射来破坏单例");
            }
        }

    }

    private volatile static ReflectDCLazyMan05 lazyMan;//加这个原因是为了防止指令重排

    public static ReflectDCLazyMan05 getInstance(){
        if(lazyMan==null){
            synchronized (ReflectDCLazyMan05.class){
                if(lazyMan==null){
                    lazyMan = new ReflectDCLazyMan05();//这个并不是一个原子性操作,它会分成三个步骤执行,这时候编译的时候会出现指令重排现象
                }
            }

        }
        return lazyMan;
    }
    public static void main(String[] args) throws Exception {
        /**
         * 下面会发现反射能创出一个不一样的实例对象
        * */
        ReflectDCLazyMan05 instance = ReflectDCLazyMan05.getInstance();
        Constructor<ReflectDCLazyMan05> declaredConstructor = ReflectDCLazyMan05.class.getDeclaredConstructor(null);//通过反射
        declaredConstructor.setAccessible(true);//它可以无视私有构造器,破除私有权限
        ReflectDCLazyMan05 instrance2 = declaredConstructor.newInstance();
        System.out.println(instance);
        System.out.println(instrance2);

    }

}

16.6单例模式使用枚举类(推荐):

Constructor<EnumSingle06> declaredConstructor = EnumSingle06.class.getDeclaredConstructor(String.class,int.class);
declaredConstructor.setAccessible(true);
EnumSingle06 instance2 = declaredConstructor.newInstance();

可以看下这个反射创建实例的源码,显示对于枚举类型,会抛出异常不能通过反射创建枚举类型:

image-20200905112356872

为什么枚举类型用作单例很好

  • 1.枚举类型是线程安全的
  • 2.枚举类型只会装载一次
  • 3.反射部分创建实例的部分源码已经写了不能用反射区创建枚举
package com.single;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

/**
 * Created by yj on 2020/9/5 11:17
 */
//enum类型是一个类
//枚举类型是所用单例实现中唯一一种不会被破坏的单例实现模式。
/**1.枚举类型是线程安全的
 * 2.枚举类型只会装载一次
 * * */
public enum EnumSingle06 {
    INSTANCE;//创建一个枚举对象,该对象天生为单例

    public EnumSingle06 getInstance(){
        return INSTANCE;
    }
}

class Test{

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        EnumSingle06 instance1 = EnumSingle06.INSTANCE;
        Constructor<EnumSingle06> declaredConstructor = EnumSingle06.class.getDeclaredConstructor(String.class,int.class);
        declaredConstructor.setAccessible(true);
        EnumSingle06 instance2 = declaredConstructor.newInstance();

        // NoSuchMethodException: com.kuang.single.EnumSingle.<init>()
        System.out.println(instance1);
        System.out.println(instance2);

    }

}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值