【Java】单例模式

单例模式是面试中常考的设计模式 之一

在面试中,面试官常常会要求写出两种类型的单例模式并解释原理

本文中,将从0到1的介绍单例模式究竟是什么

文章目录

✍一、什么是设计模式?

设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性。 毫无疑问,设计模式于己于他人于系统都是多赢的,设计模式使代码编制真正工程化,设计模式是软件工程的基石,如同大厦的一块块砖石一样。项目中合理的运用设计模式可以完美的解决很多问题,每种模式在现在中都有相应的原理来与之对应,每一个模式描述了一个在我们周围不断重复发生的问题,以及该问题的核心解决方案,这也是它能被广泛应用的原因。简单说:

模式:在某些场景下,针对某类问题的某种通用的解决方案。

场景:项目所在的环境

问题:约束条件,项目目标等

解决方案:通用、可复用的设计,解决约束达到目标。

用生活中的事务来介绍:

设计模式好⽐象棋中的 "棋谱". 红⽅当头炮, ⿊⽅⻢来跳. 针对红⽅的⼀些⾛法, ⿊⽅应招的时候有⼀些固定的套路. 按照套路来⾛局势就不会吃亏.

✍二、单例模式是什么?

单例模式是指在内存中只会创建且仅创建一次对象的设计模式 。在程序中多次使用同一个对象且作用相同时 ,为了防止频繁地创建对象使得内存飙升,单例模式可以让程序仅在内存中创建一个对象,让所有需要调用的地方都共享这一单例对象。

简单概括:

  • 单例模式能保证某个类在程序中只存在唯⼀⼀份实例,而不会创建出多个实例.

✍三、单例模式的类型

单例模式具体的实现⽅式有很多. 最常⻅的是 "饿汉" 和 "懒汉" 两种.

  • 饿汉式:在类加载过程中就创建了实例。
  • 懒汉式:在真正需要使用时,才会创建实例。

1.饿汉式

java 复制代码
class Singleton{
    private static Singleton instance = new Singleton();

    public static Singleton getInstance(){
        return instance;
    }
    private Singleton(){}
}

类在加载时,就会创建一个实例。在调用时,之间返回这一实例就好。

可以简单的认为,在程序启动时就创建了实例。

2.懒汉式

java 复制代码
class Singletonlazy{
     public static Singletonlazy instance = null;
     public Singletonlazy getInstance(){
         if ( instance == null){
             instance = new Singletonlazy();
         }
         return instance;
     }
    private Singletonlazy(){}

}

这是一段存在些许问题的代码,不过可以直观的感受到两者之间的区别。

在接下来,会对如上懒汉式代码进行优化。

3.优化懒汉式

我们先将懒汉式代码放置如下

java 复制代码
class Singletonlazy{
     public static Singletonlazy instance = null;
     public Singletonlazy getInstance(){
         if ( instance == null){
             instance = new Singletonlazy();
         }
         return instance;
     }
    private Singletonlazy(){}

}

在如上懒汉式的代码中,如果在多线程情况下,就会出现一些问题

在多线程中,线程是抢占式执行的。

那么就会给程序带来一些问题

由于线程的抢占式执行,虽说不会造成空间的浪费

但是时间的消耗确实客观存在的。

那么解决这个问题,就进行加锁操作。

java 复制代码
  public Singletonlazy getInstance(){
        synchronized (lock){
            if ( instance == null){
                instance = new Singletonlazy();
            }
        }
         return instance;
     }

这样加锁,就是将 if 和 new 打包成一个原子操作

但是这样也会出现问题

那么如何解决这个问题呢?

我们在锁的外层,在添加一个判断条件

java 复制代码
  public Singletonlazy getInstance(){
         if (instance == null) {
             synchronized (lock){
                 if ( instance == null){
                     instance = new Singletonlazy();
                 }
             }
         }
         return instance;
     }

注意:

这里的两个if条件虽然内容一样,但是意义却完全不同

  • 第一个if是判断是否要进行加锁操作
  • 第二个if是判断是否要实例创建对象

如上代码已经解决了多线程情况下的线程安全问题。

也解决了执行效率的问题。

但是还存在一个问题

指令重排

4.指令重排

概念:

为了使处理器内部的运算单元能尽量被充分利用,处理器可能会对输入的代码进行乱序执行优化,处理器会在计算之后将乱序执行的结果重组,并确保这一结果和顺序执行结果是一致的,但是这个过程并不保证各个语句计算的先后顺序和输入代码中的顺序一致。这就是指令重排序。

通俗的说,就是在不改变代码逻辑的条件下,通过更改指令的执行顺序,来达到优化代码的效果。

举例:

在这一行代码中,一个创建对象实例的过程可以在指令的角度分为三步

  1. 申请内容空间
  2. 调用构造方法(对内存空间进行初始化)
  3. 把此时内存空间的地址,赋值给instance引用

在指令重排的优化下

可能有

1 --》3 --》 2

1 --》2 --》 3

这样两种情况

复制代码
1是一定在第一步的,因为是要在保证代码逻辑的前提下,才能进行指令重排。

那么如何解决呢?

引入volatile

java 复制代码
public static volatile Singletonlazy instance = null;

使用volatile关键字修饰的变量,可以保证其指令执行的顺序与程序指明的顺序一致,不会发生顺序变换

5.完整代码

java 复制代码
class Singletonlazy{
     public static volatile Singletonlazy instance = null;
     public static Object lock = new Object();
     public Singletonlazy getInstance(){
         if (instance == null) {
             synchronized (lock){
                 if ( instance == null){
                     instance = new Singletonlazy();
                 }
             }
         }
         return instance;
     }
    private Singletonlazy(){}

}

以上就是本文所有内容,如果对你有帮助的话,点赞收藏支持一下吧!💞💞💞

相关推荐
苦夏木禾7 分钟前
js请求避免缓存的三种方式
开发语言·javascript·缓存
重庆小透明12 分钟前
力扣刷题记录【1】146.LRU缓存
java·后端·学习·算法·leetcode·缓存
超级土豆粉15 分钟前
Turndown.js: 优雅地将 HTML 转换为 Markdown
开发语言·javascript·html
lang2015092817 分钟前
Reactor操作符的共享与复用
java
TTc_27 分钟前
@Transactional事务注解的批量回滚机制
java·事务
wei_shuo1 小时前
飞算 JavaAI 开发助手:深度学习驱动下的 Java 全链路智能开发新范式
java·开发语言·飞算javaai
熊猫钓鱼>_>1 小时前
用Python解锁图像处理之力:从基础到智能应用的深度探索
开发语言·图像处理·python
GO兔1 小时前
开篇:GORM入门——Go语言的ORM王者
开发语言·后端·golang·go
欧阳秦穆2 小时前
apoc-5.24.0-extended.jar 和 apoc-4.4.0.36-all.jar 啥区别
java·jar
好开心啊没烦恼2 小时前
Python 数据分析:numpy,抽提,整数数组索引与基本索引扩展(元组传参)。听故事学知识点怎么这么容易?
开发语言·人工智能·python·数据挖掘·数据分析·numpy·pandas