【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(){}

}

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

相关推荐
森屿Serien几秒前
Spring Boot常用注解
java·spring boot·后端
轻口味27 分钟前
命名空间与模块化概述
开发语言·前端·javascript
苹果醋31 小时前
React源码02 - 基础知识 React API 一览
java·运维·spring boot·mysql·nginx
晓纪同学1 小时前
QT-简单视觉框架代码
开发语言·qt
威桑1 小时前
Qt SizePolicy详解:minimum 与 minimumExpanding 的区别
开发语言·qt·扩张策略
Hello.Reader1 小时前
深入解析 Apache APISIX
java·apache
飞飞-躺着更舒服1 小时前
【QT】实现电子飞行显示器(简易版)
开发语言·qt
明月看潮生2 小时前
青少年编程与数学 02-004 Go语言Web编程 16课题、并发编程
开发语言·青少年编程·并发编程·编程与数学·goweb
明月看潮生2 小时前
青少年编程与数学 02-004 Go语言Web编程 17课题、静态文件
开发语言·青少年编程·编程与数学·goweb
Java Fans2 小时前
C# 中串口读取问题及解决方案
开发语言·c#