Java 单例模式详细解释

目录
  • 饿汉式
  • 懒汉式
  • 懒汉式(加锁synchronized)
  • 懒汉式(部分加锁synchronized)
  • 懒汉式(DCL)
  • 懒汉式(DCL)最终版
  • 静态内部类
  • 总结

饿汉式

/**
 * 饿汉式
 * 类加载到内存后,就是实例化一个单例,JVM保证线程安全
 * 简单使用:推荐使用
 * 唯一缺点:不管用与不用,类加载时就会完成实例化
 */
public class Demo01 {
	//开始先新建一个对象
    private static final Demo01 INSTANCE = new Demo01();
	//构造
    private Demo01(){};
	//调用 getInstance 方法时返回 INSTANCE,唯一创建的对象
    public static Demo01 getInstance(){
        return INSTANCE;
    }
    public static void main(String[] args) {
        Demo01 m1 = Demo01.getInstance();
        Demo01 m2 = Demo01.getInstance();
        //结果为true
        System.out.println(m1 == m2);
    }
}
单例模式(饿汉式)优点:饿汉式是典型的空间换时间,当类装载的时候就会创建类实例,不管你用不用,先创建出来,然后每次调用的时候,就不需要再判断了,节省了运行时间。
缺点:不管用与不用,类加载时就会完成实例化,会浪费一定的内存空间
改进方法:让对象在使用的时候在进行创建。------>  懒汉式

懒汉式

/**
 * 懒汉式
 * 类加载到内存后,就是实例化一个单例,JVM保证线程不安全
 * 唯一缺点:虽然达到了按需的目的,但却带来线程不安全问题
 */
public class Demo02 {
    private static Demo02 INSTANCE ;
    private Demo02(){};
    public static Demo02 getInstance(){
        //判断 INSTANCE 是否为空
       if(INSTANCE == null){
           try{
               Thread.sleep(1);
           }catch (InterruptedException e){
               e.printStackTrace();
           }
           INSTANCE = new Demo02();
       }
       return INSTANCE;
    }
    public static void main(String[] args) {
        for(int i = 0 ; i < 100 ; i++){
            new Thread(()->
            //输出该对象的hashcode值,通过对比值是否相等来判断是不是唯一的对象
                    System.out.println(Demo02.getInstance().hashCode())
            ).start();
        }
    }
}
单例模式(懒汉式)优点:懒汉式是典型的时间换空间,也就是每次获取实例都会进行判断,看是否需要创建实例,浪费判断的时间。当然,如果一直没有人使用的话,那就不会创建实例,则节约内存空间。
缺点:懒汉式在多个线程进行访问时有可能会出现多个不同的对象。
改进方法:对创建方法getInstance加锁    ------>   懒汉式(加锁synchronized)

懒汉式(加锁synchronized)

/**
 * 懒汉式(加锁)
 * 类加载到内存后,就是实例化一个单例,给创建对象的方法的加锁,JVM保证线程安全
 * 唯一缺点:虽然加锁之后可以保证线程是安全的,但会使得整个方法变慢。
 */
public class Demo03 {
    private static Demo03 INSTANCE ;
    private Demo03(){};
    //方法加锁
    public static synchronized Demo03 getInstance(){
        //业务逻辑
        //判断 INSTANCE 是否为空
       if(INSTANCE == null){
           try{
               Thread.sleep(1);
           }catch (InterruptedException e){
               e.printStackTrace();
           }
           INSTANCE = new Demo03();
       }
       return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) {
        for(int i = 0 ; i < 100 ; i++){
            new Thread(()->
                    System.out.println(Demo03.getInstance().hashCode())
            ).start();
        }
    }
}
单例模式(懒汉式(加锁))优点:懒汉式(加锁)可以保证线程的安全性,但是当上锁的方法getInstance中存在业务逻辑代码时,会拉低整个对象创建过程中速度。
缺点:对整个方法加锁,降低了方法运行的时间
改进方法:对创建方法的程序块进行上锁,业务逻辑代码部分不上锁  -------->懒汉式(部分加锁synchronized)

懒汉式(部分加锁synchronized)

/**
 * 懒汉式(部分加锁)
 * 类加载到内存后,就是实例化一个单例,给创建对象的方法的部分加锁,降低时间
 */
public class Demo04 {
    private static Demo04 INSTANCE ;
    private Demo04(){};
    //方法加锁
    public static Demo04 getInstance(){
        //业务逻辑
        //判断 INSTANCE 是否为空
       if(INSTANCE == null){
           //对方法的部分代码块进行上锁
           synchronized (Demo04.class){
               try{
                   Thread.sleep(1);
               }catch (InterruptedException e){
                   e.printStackTrace();
               }
               INSTANCE = new Demo04();
           }
       }
       return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) {
        for(int i = 0 ; i < 100 ; i++){
            new Thread(()->
                    System.out.println(Demo04.getInstance().hashCode())
            ).start();
        }
    }
}
单例模式(部分加锁懒汉式)优点:加快了程序的运行,只对创建对象的部分进行加锁
缺点:通过if判断后会有多个线程在等待线程资源,等第一个线程执行完成后还会进行第二个线程创建对象。
改进方法:加入两层if判断可以防止该问题出现 -------->双层检查锁

懒汉式(DCL)

/**
 * 懒汉式(DCL)
 * Double Check Lock
 */
public class Demo04 {
    private static Demo04 INSTANCE ;
    private Demo04(){};
    //方法加锁
    public static Demo04 getInstance(){
        //业务逻辑
        //判断 INSTANCE 是否为空
       if(INSTANCE == null){
           //对方法的部分代码块进行上锁
           synchronized (Demo04.class){
               //再次进行判断,检查 INSTANCE 是否为空
               if(INSTANCE == null){
                   try{
                       Thread.sleep(1);
                   }catch (InterruptedException e){
                       e.printStackTrace();
                   }
               }
               INSTANCE = new Demo04();
           }
       }
       return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) {
        for(int i = 0 ; i < 100 ; i++){
            new Thread(()->
                    System.out.println(Demo04.getInstance().hashCode())
            ).start();
        }
    }
}
单例模式(懒汉式DCL)优点:加快了对象创建的时间,同时保证了线程的安全性。
缺点:当对象发生指令重排时,第二个线程虽然拿到了对象,但是是拿到的不完整的对象,容易出现问题
改进方法:给该方法加上volatile关键字进行上锁可以防止指令重排问题。

延伸一下:为什么要用两层if判断呢?

答:因为使用两层if可以提高方法的运行速度,因为if判断消耗的时间较少,但是synchronized 消耗的时间却很大。在外面加上一层if,可以帮助过滤掉很多线程访问。

懒汉式(DCL)最终版

/**
 * 懒汉式(DCL)
 * Double Check Lock
 */
public class Demo04 {
    private static volatile Demo04 INSTANCE ;
    private Demo04(){};
    //方法加锁
    public static Demo04 getInstance(){
        //业务逻辑
        //判断 INSTANCE 是否为空
       if(INSTANCE == null){
           //对方法的部分代码块进行上锁
           synchronized (Demo04.class){
               //再次进行判断,检查 INSTANCE 是否为空
               if(INSTANCE == null){
                   try{
                       Thread.sleep(1);
                   }catch (InterruptedException e){
                       e.printStackTrace();
                   }
               }
               INSTANCE = new Demo04();
           }
       }
       return INSTANCE;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) {
        for(int i = 0 ; i < 100 ; i++){
            new Thread(()->
                    System.out.println(Demo04.getInstance().hashCode())
            ).start();
        }
    }
}

对 INSTANCE 进行上锁可以防止指令重排,保证对象的完整性。

延伸:DCL模式为什么要加上volatile ?

答:我们要从java对象创建过程和CPU乱序执行两个方面考虑。

java对象创建过程可分为:

1:内存中分配空间
2:初始化对象
3:变量与对象关联

当发生指令重排是顺序变为

1:内存中分配空间
3:变量与对象关联
2:初始化对象

第一个线程访问时,发生指令重排,对象刚创建一半,还未对对象内部的值进行初始化赋值。此时第二个线程进行访问,此时他读取到的就是创建到一半的对象,初始化为空的对象。最终就会导致对象不完整。

静态内部类

加载外部类时不会加载内部类,只有第一次调用getInstance方法时,JVM才加载 Singleton04Holder 并初始化INSTANCE ,只有一个线程可以获得对象的初始化锁,其他线程无法进行初始化,保证对象的唯一性。

public class Demo04 {
    private Demo04 () {
    }
    private static class Demo04Holder {
        private final static Demo04 INSTANCE = new Demo04 ();
    }
    public static Demo04 getInstance() {
        return Demo04Holder.INSTANCE;
    }
    public static void main(String[] args) {
        for(int i=0; i<100; i++) {
            new Thread(()->{
                System.out.println(Demo04.getInstance().hashCode());
            }).start();
        }
    }
}

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • 深入理解Java设计模式之单例模式

    目录 一.什么是单例模式 二.单例模式的应用场景 三.单例模式的优缺点 四.单例模式的实现 1.饿汉式 2.懒汉式 3.双重加锁机制 4.静态初始化 五.总结 一.什么是单例模式 单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在. 许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为.比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息.这种方式简

  • Java单例模式的6种实现方式详解

    目录 为什么使用单例模式 使用单例模式需要注意的关键点 单例模式的几种写法 1. 饿汉式 2. 懒汉式 3. DCL(Double CheckLock)实现单例 4. 静态内部类 5. 枚举单例 6. 容器实现单例 总结 为什么使用单例模式 需要确保某个类只要一个对象,或创建一个类需要消耗的资源过多,如访问IO和数据库操作等,这时就需要考虑使用单例模式了. 使用单例模式需要注意的关键点 将构造函数访问修饰符设置为private 通过一个静态方法或者枚举返回单例类对象 确保单例类的对象有且只有

  • Java中的单例模式详解(完整篇)

    目录 前言 WHAT WHY 饿汉式 实现一:静态实例参数与静态代码块 实现二:静态内部类 懒汉式 错误一:单线程实现 错误二:同步方法 错误三:同步代码块之单次检查 错误四:同步代码块之双重检查 正确:双重检查+阻止重排序 枚举 场景 总结 前言 个人认为单例模式是设计模式中最简单也是最常用的一种,是对有限资源合理利用的一种方式.这个模式看似简单,但是其中蕴含了关于并发.类加载.序列化等一系列深层次的知识,如果理解不够深,就有可能在高并发时遇到难以预期的异常,或者会造成资源浪费. 所以本文会从

  • Java单例模式分析

    目录 单例模式 为什么要用单例 单例的关键点 几种写法 懒汉式 饿汉式 静态内部类写法 枚举单例 容器实现单例 参考 总结 单例模式 为什么要用单例 确保某个类只有一个对象,常用于访问数据库操作,服务的配置文件等. 单例的关键点 1.默认构造函数为private,复制构造函数和复制赋值函数也要private或=delete禁用.(做到无法被外部其他对象构造) 2.通过一个静态方法或枚举返回单例类对象. 3.确保多线程的环境下,单例类对象只有一个. 几种写法 本文主要介绍C++的懒汉式和饿汉式写法

  • 浅析Java单例设计模式(自写demo)

    目录 单例模式特点 单例模式优点 实现方式 饿汉式(线程安全) 懒汉式 单例模式特点 1.构造器私有 2.在一个Java应用程序中,可保证只有一个实例对象 3.只提供一个供外界调用的getInstance()方法 单例模式优点 1.减少某些对象的频繁创建,降低系统开销和内存占用 2.外部调用不使用new关键字,降低系统内存的使用频率 3.对于特殊的类,在系统中只能存在一个实例,否则系统无法正常运行,比如Controller 实现方式 这里简单介绍两种实现方式 饿汉式(线程安全) /** * @a

  • Java 实例解析单例模式

    目录 单例模式的介绍 优点 缺点 Synchronized Synchronized示例 Synchronized与非Synchronized Singleton 第一个示例 第二个示例 第三个示例 第四个示例 第五个示例 单例模式的介绍 单例对象(Singleton)是一种常用的设计模式.在实际使用中,单例对象能保证在一个JVM中,该对象只存在一个实例存在. 优点 1.减少系统开销,提高系统性能 2.省去了new操作符,降低了系统内存的使用频率,减轻GC压力 3.避免对共享资源的多重占用 缺点

  • Java 单例模式详细解释

    目录 饿汉式 懒汉式 懒汉式(加锁synchronized) 懒汉式(部分加锁synchronized) 懒汉式(DCL) 懒汉式(DCL)最终版 静态内部类 总结 饿汉式 /** * 饿汉式 * 类加载到内存后,就是实例化一个单例,JVM保证线程安全 * 简单使用:推荐使用 * 唯一缺点:不管用与不用,类加载时就会完成实例化 */ public class Demo01 { //开始先新建一个对象 private static final Demo01 INSTANCE = new Demo0

  • Java超详细分析垃圾回收机制

    目录 前言 垃圾回收概述 内存溢出和内存泄漏 垃圾回收算法 标记阶段 STW(Stop-the-World) 回收阶段 标记-清除算法 复制算法 标记-压缩算法 三种算法的比较 总结 前言 在前面我们对类加载, 运行时数据区 ,执行引擎等作了详细的介绍 , 这节我们来看另一重点 : 垃圾回收. 垃圾回收概述 垃圾回收是java的招牌能力 ,极大的提高了开发效率, java是自动化的垃圾回收, 其他语言有的则需要程序员手动回收 , 那么什么是垃圾呢? 垃圾是指在运行程序中没有任何引用指向的对象,这

  • cmd 环境变量设置方法详细解释

    cmd设置环境变量可以方便我们bat脚本的运行,但是要注意的是变量只在当前的cmd窗口有作用(局部生效),如果想要设置持久的环境变量需要我们通过两种手段进行设置:一种是直接修改注册表,另一种是通过我的电脑-〉属性-〉高级,来设置系统的环境变量. 1.查看当前所有可用的环境变量:输入 set 即可查看. 2.查看某个环境变量:输入 "set 变量名"即可,比如想查看temp变量的值,即输入 set temp 当然也可以使用echo %temp% 3.修改环境变量 :输入 "se

  • java单例模式实现的方法

    1.最基本的单例模式 /** * @author LearnAndGet * @time 2018年11月13日 * 最基本的单例模式 */public class SingletonV1 { private static SingletonV1 instance = new SingletonV1();; //构造函数私有化 private SingletonV1() {} public static SingletonV1 getInstance() { return instance; }

  • java 单例模式容易忽略的细节

    java单例模式 直接讲实现单例模式的两种方法:懒汉式和饿汉式,单例模式的概念自己上网搜吧这里就不讲了! 这里会涉及到java中的jvm,如果你没有这方面的知识,我建议你先去补补,不然会有点迷糊! 首先说说类什么时候进行加载? java虚拟机没有进行强制性的约束,但是对于初始化却严格规定了有且只有4种情况必须先对类进行初始化. 我们要知道的是在类加载的过程中,加载.验证.准备是在初始化之前完成的,所以进行了初始化,加载.验证.准备自然就在之前完成了. 然后这四种情况是分别遇到 new . get

  • Java超详细分析泛型与通配符

    目录 1.泛型 1.1泛型的用法 1.1.1泛型的概念 1.1.2泛型类 1.1.3类型推导 1.2裸类型 1.3擦除机制 1.3.1关于泛型数组 1.3.2泛型的编译与擦除 1.4泛型的上界 1.4.1泛型的上界 1.4.2特殊的泛型上界 1.4.3泛型方法 1.4.4类型推导 2.通配符 2.1通配符的概念 2.2通配符的上界 2.3通配符的下界 题外话: 泛型与通配符是Java语法中比较难懂的两个语法,学习泛型和通配符的主要目的是能够看懂源码,实际使用的不多. 1.泛型 1.1泛型的用法

  • Java超详细讲解设计模式之一的工厂模式

    目录 工厂模式 1.简单工厂 1.1结构 1.2实现 1.3优缺点 1.4扩展 2.工厂方法 2.1结构 2.2实现 2.3优缺点 3.抽象工厂 3.1结构 3.2实现 3.3优缺点 4.模式扩展 4.1实现 工厂模式 在Java应用程序中对象无处不在,这些对象都需要进行创建,如果创建的时候直接new对象,那么如果我们要更换对象,所有new对象的地方都需要进行更改.违背了软件设计原则中的开闭原则.如果我们使用工厂生产对象,只需要在工厂中关注对象的改变即可,达到了与对象解耦的目的,工厂模式最大的特

  • Java 超详细讲解类的定义方式和对象的实例化

    目录 1.面对对象的初步认识 1.1什么是面向对象 1.2面向对象与面向过程 2.类的定义与使用 2.1简单认识类 2.2 类的定义格式 3.类的实例化 3.1什么是实例化? 3.2重点笔记 总结 1.面对对象的初步认识 1.1什么是面向对象 用面向对象的思想来涉及程序,更符合人们对事物的认知,对于大型程序的设计.扩展以及维护都非常友好. 1.2面向对象与面向过程 举一个买手机的例子 以面向对象的方式来处理买手机这件事的话,我们就不需要关注买手机的过程,具体手机怎么买,如何到手,用户不用去关心,

  • Java 超详细讲解十大排序算法面试无忧

    目录 排序算法的稳定性: 一.选择排序 二.冒泡排序 三.插入排序 四.希尔排序 五.堆排序 六.归并排序 七.快速排序 八.鸽巢排序 九.计数排序 十.基数排序 排序算法的稳定性: 假定在待排序的记录序列中,存在多个具有相同的关键字的记录,如果排序以后,保证这些记录的相对次序保持不变,即在原序列中,a[i]=a[j],且 a[i] 在 a[j] 之前,排序后保证 a[i] 仍在 a[j] 之前,则称这种排序算法是稳定的:否则称为不稳定的. 一.选择排序 每次从待排序的元素中选择最小的元素,依次

  • Java 超详细讲解类的定义方式和对象的实例化

    目录 1.面对对象的初步认识 1.1什么是面向对象 1.2面向对象与面向过程 2.类的定义与使用 2.1简单认识类 2.2 类的定义格式 3.类的实例化 3.1什么是实例化? 3.2重点笔记 总结 1.面对对象的初步认识 1.1什么是面向对象 用面向对象的思想来涉及程序,更符合人们对事物的认知,对于大型程序的设计.扩展以及维护都非常友好. 1.2面向对象与面向过程 举一个买手机的例子 以面向对象的方式来处理买手机这件事的话,我们就不需要关注买手机的过程,具体手机怎么买,如何到手,用户不用去关心,

随机推荐