当前位置: 首页 > news >正文

单例模式 之内部类延迟加载,(多)线程安全

单例模式,很多种方式实现,但是这儿只介绍最优方案。

就是利用内部类去实现单例模式。

这种单例模式的好处就是,延迟加载,减少内存开销,访问成本低且线程安全。

直接上代码:

/**
 * @Author : JCccc
 * @CreateTime : 2018-11-5
 * @Description :
 * @Point: Keep a good mood
 **/

public class SingletonInner {




//    静态嵌套类(Static Nested Class)和内部类(Inner Class)的不同
//    静态嵌套类是被声明为静态(static)的内部类,可以不依赖于外部类被实例化,
//    内部类需要在外部类实例化后才能实例化.


    /**
     * 内部类实现单例模式
     * 延迟加载,减少内存开销
     */

    /**
     * 在内部类SingletonHolder 第一次被加载的时候,创建了instance,指向SingletonInner的实例。
     */
    private static class SingletonHolder {
        private static SingletonInner instance = new SingletonInner();
    }

    /**
     * 私有的构造函数
     */
    private SingletonInner() {
        System.out.println("Singleton is create");
    }

    public static SingletonInner getInstance() {

        return SingletonHolder.instance;
    }

    public void method() {
        System.out.println("SingletonInner!");
    }



    public static void main(String[] args) {
        SingletonInner Single_dog1 = SingletonInner.getInstance();
        SingletonInner Single_dog2 = SingletonInner.getInstance();
        SingletonInner Single_dog3 = SingletonInner.getInstance();
        System.out.println(Single_dog1+"\n"+Single_dog2+"\n"+Single_dog3);


        


    }
}

 针对内部类为何能保证线程安全,实现延迟加载,下面是简要的说明:

   private static class SingletonHolder {
        private static SingletonInner instance = new SingletonInner();
    }

 

     这个内部类因为被static修饰,代表,这个货是一个类级别的内部类。
     如果没有被static修饰,就是一个对象级别的内部类,这种内部类是必须绑定在外部对象实例上的
     这种类级别内部类的好处是什么,就是虚拟机JVM的内部机制对它是有保护的,只允许它第一次被加载,其余都是互斥的。
     

   虚拟机会保证一个类的类构造器<clinit>()在多线程环境中被正确的加锁、同步,如果多个线程同时去初始化一个类,那么只会     有一个线程去执行这个类的类构造器<clinit>(),其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕。

   特别需要注意的是,在这种情形下,其他线程虽然会被阻塞,但如果执行<clinit>()方法的那条线程退出后,其他线程在唤醒之     后不会再次进入/执行<clinit>()方法,因为 在同一个类加载器下,一个类型只会被初始化一次。如果在一个类的<clinit>()方法中     有耗时很长的操作,就可能造成多个线程阻塞,在实际应用中这种阻塞往往是隐藏的。

 

 

这里其实是验证小测试,看看运行结果,就懂了吧:

没错,就是单例验证。 也许你debug去调试,你会发现创建了3次对象,实际上都是指向同一个的。

好了,到此。 

相关文章:

  • Springboot 用session监听器统计在线用户数量
  • (Redis使用系列) Springboot 实现Redis消息的订阅与分布 四
  • JAVA 责任链设计模式
  • JAVA回调函数简单讲解 CallBack
  • JAVA HttpClient 远程调用接口doGet、doPost工具类
  • JAVA 接口签名sign生成 工具类
  • mybatis-config配置文件各项简单介绍
  • Springboot整合Mybatis增删查改、连接MYSQL数据库及配置druid连接池
  • JAVA 模板设计模式
  • Springboot 最简单的结合MYSQL数据实现EXCEL表格导出及数据导入
  • Springboot中使用GSON报错 An attempt was made to call the method com.google.gson.GsonBuilder.setLenient
  • IDEA @AutoWired注入bean 出现红色波浪线
  • JAVA 最常用实用的正则表达式校验
  • Springboot 整合WebFlux 实现RESTFUI风格API 及简单的CRUD
  • Springboot 读取配置文件application.properties (yml)的四种方式
  • 11111111
  • CSS居中完全指南——构建CSS居中决策树
  • iOS帅气加载动画、通知视图、红包助手、引导页、导航栏、朋友圈、小游戏等效果源码...
  • Lsb图片隐写
  • python3 使用 asyncio 代替线程
  • vue-router的history模式发布配置
  • Web Storage相关
  • Yeoman_Bower_Grunt
  • 阿里云购买磁盘后挂载
  • 多线程事务回滚
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 开源SQL-on-Hadoop系统一览
  • 前端
  • 问:在指定的JSON数据中(最外层是数组)根据指定条件拿到匹配到的结果
  • #中的引用型是什么意识_Java中四种引用有什么区别以及应用场景
  • %3cscript放入php,跟bWAPP学WEB安全(PHP代码)--XSS跨站脚本攻击
  • (pojstep1.3.1)1017(构造法模拟)
  • (python)数据结构---字典
  • (删)Java线程同步实现一:synchronzied和wait()/notify()
  • (十一)手动添加用户和文件的特殊权限
  • (五)c52学习之旅-静态数码管
  • (转)EXC_BREAKPOINT僵尸错误
  • 、写入Shellcode到注册表上线
  • .Net 8.0 新的变化
  • .NET Core 网络数据采集 -- 使用AngleSharp做html解析
  • .NET Framework与.NET Framework SDK有什么不同?
  • .NET 使用 ILRepack 合并多个程序集(替代 ILMerge),避免引入额外的依赖
  • .netcore 获取appsettings
  • .NetCore项目nginx发布
  • /*在DataTable中更新、删除数据*/
  • /etc/apt/sources.list 和 /etc/apt/sources.list.d
  • /etc/sudoer文件配置简析
  • ?php echo ?,?php echo Hello world!;?
  • @31省区市高考时间表来了,祝考试成功
  • [ 渗透工具篇 ] 一篇文章让你掌握神奇的shuize -- 信息收集自动化工具
  • [<MySQL优化总结>]
  • [2016.7 test.5] T1
  • [⑧ADRV902x]: Digital Pre-Distortion (DPD)学习笔记
  • [C#][opencvsharp]opencvsharp sift和surf特征点匹配
  • [C#小技巧]如何捕捉上升沿和下降沿