Java单例---静态内部类

之前写过一篇双重锁校验单例 ,这是延迟加载的一种单例模式,俗称懒汉模式,这次写一个静态内部类的单例,这个写法天生线程安全,并且引出反射攻击,废话不多说,上代码:

java 复制代码
public class StaticInnerClass {

    private static class InnerClass{
        private static StaticInnerClass staticInnerClass = new StaticInnerClass();
    }

    public static StaticInnerClass getInstance(){
        return InnerClass.staticInnerClass;
    }

    private StaticInnerClass(){
    	
    }

}

这种静态内部类的实现方式,它主要原理是:加载一个类时,其内部类不会同时被加载。一个类被加载,当且仅当其某个静态成员(静态域、构造器、静态方法等)被调用时发生 ,也就是说内部类:InnerClass只有我们调用getInstance()的时候才会被加载。

那么这个写法为什么线程安全呢?原因可以在《深入理解Java虚拟机》这本书的第七章7.3.5小节找到答案,这里摘要出主要原因,大家有兴趣可以自己去看看书:

  • 虚拟机会保证一个类的< clinit>()方法在多线程环境中被正确的加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的< clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行< clinit>()方法完毕。如果在一个类的< clinit>()方法中有耗时很长的操作,就有可能造成多个进程阻塞(需要注意的是,其他线程虽然会被阻塞,但如果执行< clinit>()方法的那条线程退出< clinit>()方法后,其他线程唤醒之后不会再次进入< clinit>()方法。同一个类加载器下,一个类型只会被初始化一次),在实际应用中这种阻塞旺旺是很隐蔽的。

这就是他线程安全的原因,通俗的说就是这个静态内部类只能同时有一个线程去初始化它,其他的被阻塞,如果一个线程初始化了这个静态内部类,那么其他的就不会再去初始化了。

那么上面这个写法就真的能保证这个类的实例在任何情况下都只有一个吗?其实是不行的,通过反射的方式,就可以修改这个类的私有构造器权限,然后创建出一个这个类的实例,这个在下一篇博客里面写。

个人浅薄理解,欢迎补充

相关推荐
时差9538 分钟前
【面试题】Hive 查询:如何查找用户连续三天登录的记录
大数据·数据库·hive·sql·面试·database
让学习成为一种生活方式10 分钟前
R包下载太慢安装中止的解决策略-R语言003
java·数据库·r语言
晨曦_子画16 分钟前
编程语言之战:AI 之后的 Kotlin 与 Java
android·java·开发语言·人工智能·kotlin
假装我不帅36 分钟前
asp.net framework从webform开始创建mvc项目
后端·asp.net·mvc
南宫生39 分钟前
贪心算法习题其三【力扣】【算法学习day.20】
java·数据结构·学习·算法·leetcode·贪心算法
神仙别闹39 分钟前
基于ASP.NET+SQL Server实现简单小说网站(包括PC版本和移动版本)
后端·asp.net
Heavydrink1 小时前
HTTP动词与状态码
java
ktkiko111 小时前
Java中的远程方法调用——RPC详解
java·开发语言·rpc
计算机-秋大田1 小时前
基于Spring Boot的船舶监造系统的设计与实现,LW+源码+讲解
java·论文阅读·spring boot·后端·vue
神里大人1 小时前
idea、pycharm等软件的文件名红色怎么变绿色
java·pycharm·intellij-idea