IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 【设计模式】Java单例模式-深度分析 -> 正文阅读

[Java知识库]【设计模式】Java单例模式-深度分析

饿汉式,一上来就创建

//饿汉式单例
public class Hungry {

    private Hungry(){

    }
    private final static Hungry HUNGRY= new Hungry();
    public static Hungry getInstance() {
        return HUNGRY;
    }
    //饿汉式可能浪费空间
}

懒汉式,用的才创建

直接这样写,在多线程并发时会有问题,可能导致不是单例的

public class LazyMan {

    private  LazyMan() {
        System.out.println(Thread.currentThread().getName() + "ok");
    }

    private static LazyMan lazyMan;

    public static LazyMan getInstance() {
        //懒汉式,用到才创建, 但是在多线程并发时有问题!
        if(lazyMan == null) {
            lazyMan = new LazyMan();
        }
        return lazyMan;
    }

    //模拟多线程并发
    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                LazyMan.getInstance();
            }).start();

        }
    }
}

加双重检测锁,volatile 关键字,保证懒汉单例创建

为什么进行双重检测而不是直接在方法上加synchronized?

在方法上加synchronized可以保证单例创建对象,但是对象创建完了,每次都只能一个线程获取对象,显然不合理
第一个检测 lazyMan == null是为了判断对象是否实例化,如果以及实例化了,直接返回对象即可。只有未实例化时才进去竞争锁
第二个检测是因为可能有两个线程都进入资源竞争,第一个线程创建完对象,释放锁,第二个可能继续又创建一个出来违反单例规则

为什么加 volatile 关键字?

因为new 对象这个操作不是原子操作,分为三步:1.分配内存空间, 2.执行构造方法初始化对象, 3.把这个对象指向空间
而且这三步操作在底层可能会出现指令重排, 导致执行顺序并不是123,可能是132。
这样可能会A线程先执行了3对象指向空间不为空了,还没执行2, 然后线程B进来发现对象不为空直接返回还没初始化好的对象,出现错误
volatile 关键字有禁止指令重排的作用

public class LazyMan {

    private  LazyMan() {
        System.out.println(Thread.currentThread().getName() + "ok");
    }

    private volatile static LazyMan lazyMan;

    public static LazyMan getInstance() {
        //懒汉式,用到才创建, 但是在多线程并发时有问题!
        //双重检测锁
        if(lazyMan == null) {
            synchronized (LazyMan.class) {
                if(lazyMan == null) {
                    lazyMan = new LazyMan();
                    // new 不是一个原子操作, 1.分配内存空间, 2.执行构造方法初始化对象, 3.把这个对象指向空间
                    // 底层是这3步操作,且可能出现指令重排,我们期望123执行,实际可能132执行
                    // 如果线程A先执行了3, 把对象指向了空间,空间没值,线程B来的时候对象 != null了,导致出现错误, 所以对象得加 volatile 包含禁止指令重排序的含义
                }
            }
        }
        return lazyMan;
    }

    //模拟多线程并发
    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                LazyMan.getInstance();
            }).start();
        }
    }
}

内部类

内部类实现的单例是懒加载的,且线程安全
静态内部类只有在第一次使用的时候才会被加载,由JVM保证其线程安全性,确保该成员变量只能初始化一次

补充:静态属性和静态代码块都是在类加载的时候进行初始化和执行,两者优先级一样,按编码顺序初始化
非静态属性和非静态代码块在构造方法之前执行
静态变量、静态代码块、 非静态变量、普通代码块 执行完毕后执行构造方法
属性加载顺序回顾可以看文章最后一章

public class Holder {
    private Holder(){

    }
    public static Holder getInstance() {
        return InnerClass.HOLDER;
    }
    public static class InnerClass {
        private static final Holder HOLDER = new Holder();
    }
}

用反射破坏单例及相应对抗方式

使用单例破坏构造器的私有性

    public static void main(String[] args) throws Exception {
        LazyMan instance1 = LazyMan.getInstance();
   		//通过反射拿到无参构造器
        Constructor<LazyMan> constructor = LazyMan.class.getDeclaredConstructor(null);
        constructor.setAccessible(true); //破坏构造器私有
        LazyMan instance2 = constructor.newInstance();
        System.out.println(instance1 == instance2); //false
    }

防止反射的对抗, 构造器方法:

    private  LazyMan() {
        synchronized (LazyMan.class) {
            if(lazyMan != null) {
                throw new RuntimeException("不要试图使用反射破坏单例");
            }
        }
    }

用反射先获取无参构造器, 导致 if(lazyMan == null) 条件失效, 还是一直可以创建对象

    public static void main(String[] args) throws Exception {
   		//通过反射拿到无参构造器
        Constructor<LazyMan> constructor = LazyMan.class.getDeclaredConstructor(null);
        constructor.setAccessible(true);
        LazyMan instance2 = constructor.newInstance();
        LazyMan instance3 = constructor.newInstance();
        System.out.println(instance2 == instance3); //false
    }

再次防止反射的对抗, 创建一个静态变量作为标志位,构造器方法:

//创建一个谁都不知道的静态变量, 作为判断条件
    private static boolean lych4 = false;
    private  LazyMan() {
        synchronized (LazyMan.class) {
            if(!lych4) {
                lych4 = true;
            }else {
                throw new RuntimeException("不要试图使用反射破坏单例");
            }
        }
    }

假设用某种方法获取到了你标志位变量的名字,然后用反射获取到这个属性,破坏其私有,手动改变其值,破坏单例

    public static void main(String[] args) throws Exception {

        Field lych4 = LazyMan.class.getDeclaredField("lych4");
        lych4.setAccessible(true); //破坏属性的私有性

        Constructor<LazyMan> constructor = LazyMan.class.getDeclaredConstructor(null);
        constructor.setAccessible(true);
        LazyMan instance1 = constructor.newInstance();
        lych4.set(instance1, false); //设置lych4的值
        LazyMan instance2 = constructor.newInstance();

        System.out.println(instance1 == instance2);
    }

到底怎么办呢?

真正解决,实现防止被破坏的单例

先看反射的构造器 newInstance() 方法源码
在这里插入图片描述
可以发现如果类是一个枚举类(ENUM)的话,会抛出异常:“不能用反射创建枚举对象”

所有如果单例模式不想被反射破坏,终极武器就是使用枚举对象

测试:

创建枚举类

/**
 * enum 本身也是一个class类
 */
public enum  EnumSingle {

    INSTANCE;

    public  EnumSingle getInstance() {
        return INSTANCE;
    }
}

尝试反射破坏枚举类, 查看EnumSingle 的class文件发现里面有一个空参构造器,
尝试通过反射获取空参构造器去创建对象,

    public static void main(String[] args) throws Exception {
        EnumSingle instance1 = EnumSingle.INSTANCE;
        Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(null);
        declaredConstructor.setAccessible(true);
        EnumSingle instance2 = declaredConstructor.newInstance();
        System.out.println(instance1 == instance2);
    }

报错:Exception in thread “main” java.lang.NoSuchMethodException: cn.lych4.sigle.EnumSingle.()
报错说没有这个方法(构造器), 而不是报的 “不能用反射创建枚举对象”。不太对劲,这说明枚举类里面没有空参构造器。class文件显示的Java代码不太对

经过专业软件,把class文件转为Java文件发现枚举类里面是有参构造

发现有参构造,那我们通过反射获取有参构造器,尝试用反射破坏单例

    public static void main(String[] args) throws Exception {
        EnumSingle instance1 = EnumSingle.INSTANCE;
        Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(String.class, int.class);
        declaredConstructor.setAccessible(true);
        EnumSingle instance2 = declaredConstructor.newInstance();
        System.out.println(instance1 == instance2);
    }

报错:Cannot reflectively create enum objects at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
成功报了:“不能用反射创建枚举对象”这个错误,说明确实不能用反射来破坏枚举类的单例。

如果发现有错误的地方,欢迎大家提出批评指正

💖致力于分享记录各种知识干货,关注我,让我们一起进步,互相学习,不断创作更优秀的文章
💖💖不要忘了三连哦 👍 💬 ?? , 会回访的

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-06-06 17:11:36  更:2022-06-06 17:12:54 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 20:08:56-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码