手撕单例的五种写法!

开发 前端
单例模式是一种常见的设计模式,它确保一个类只有一个实例,并提供一个全局访问点来获取该实例。当然,它也是面试中的常客,尤其是某手面试中经常要求应聘者手撕,所以今天咱们就来盘盘它。

单例模式是一种常见的设计模式,它确保一个类只有一个实例,并提供一个全局访问点来获取该实例。当然,它也是面试中的常客,尤其是某手面试中经常要求应聘者手撕,所以今天咱们就来盘盘它。

单例模式的实现方式有很多,如下图所示:

具体实现如下。

1.饿汉式模式

此在饿汉式单例模式中,实例在类加载时就被创建,这种方式的优点是实现简单,线程安全(因为类加载过程是线程安全的)。缺点是可能会导致实例过早创建,如果实例创建过程比较耗时或者占用大量资源,而在程序运行初期并不需要该实例,就会造成资源浪费。

public class Singleton {
    // 1.私有静态成员变量,在类加载时就创建实例
    private static Singleton instance = new Singleton();

    // 2.私有构造函数,防止外部通过构造函数创建实例
    private Singleton() {}

    // 3.公共静态方法,用于获取唯一的实例
    public static Singleton getInstance() {
        return instance;
    }
}

2.懒汉模式(非安全)

懒汉式单例模式在第一次调用 getInstance 方法时才创建实例,这样可以避免实例过早创建。但上述代码是非线程安全的,在多线程环境下,可能会出现多个线程同时进入 if 语句,导致创建多个实例的情况。

public class Singleton {
    // 1.私有静态成员变量,初始化为null
    private static Singleton instance = null;

    // 2.私有构造函数,防止外部通过构造函数创建实例
    private Singleton() {}

    // 3.公共静态方法,用于获取唯一的实例
    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

3.懒汉模式(安全效率低)

此版本的懒汉式单例模式通过在 getInstance 方法上添加 synchronized 关键字,使其成为线程安全的。但这种方式的缺点是每次调用 getInstance 时都需要获取锁,会导致性能下降,尤其是在高并发环境下。

public class Singleton {
    // 1.私有静态成员变量,初始化为null
    private static Singleton instance = null;

    // 2.私有构造函数,防止外部通过构造函数创建实例
    private Singleton() {}

    // 3.公共静态方法,用于获取唯一的实例
    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

4.双重检查锁模式

双重检查锁定模式在懒汉式基础上进行了优化,通过两次检查 instance 是否为 null,既保证了在第一次需要实例时创建实例,又在一定程度上避免了每次调用 getInstance 都获取锁的情况,提高了性能。不过,由于指令重排序等问题,可能会导致一些错误,因此需要在 instance 变量前添加 volatile 关键字来解决。

public class Singleton {
    // 1.私有静态成员变量,初始化为null
    private volatile static Singleton instance = null;

    // 2.私有构造函数,防止外部通过构造函数创建实例
    private Singleton() {}

    // 3.公共静态方法,用于获取唯一的实例
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

5.静态内部类模式

这种方式利用了静态内部类的特性,当外部类被加载时,静态内部类不会被加载,只有当调用 getInstance 方法时,静态内部类才会被加载,此时才创建单例实例。这种实现方式既保证了线程安全,又避免了在不需要实例时过早创建实例,是一种比较常用的单例模式实现方式。

public class Singleton {
    // 1.私有构造函数,防止外部通过构造函数创建实例
    private Singleton() {}

    // 2.静态内部类,其中包含单例实例
    private static class SingletonHolder {
        private static final Singleton instance = new Singleton();
    }

    // 3.公共静态方法,用于获取唯一的实例
    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }
}

小结

单例模式虽然实现方式有 5  种:饿汉模式、懒汉非安全模式、懒汉安全模式、双重较验锁模式、静态内部类模式,但它的写法基本都是以下三步:

  • 定义私有构造方法(防止 new 多个实例)。
  • 定义私有变量(承接单例对象)。
  • 定义统一返回对象的方法。
责任编辑:姜华 来源: 磊哥和Java
相关推荐

2022-08-10 11:02:56

Python单例模式

2023-11-13 16:49:51

C++单例

2022-05-23 07:35:15

单例模式懒汉模式静态内部类

2021-07-15 14:29:06

LRU算法

2021-09-04 07:56:44

单例模式写法

2021-09-06 08:13:35

APM系统监控

2021-06-10 09:00:33

单例模式数据库

2020-09-17 14:04:32

拷贝

2020-09-15 08:55:07

算法数据基础

2018-04-03 15:38:07

Java单例模式模式设计

2020-09-16 14:17:42

flat方法

2021-10-31 07:38:37

排序算法代码

2023-09-18 09:10:11

Golang高性能缓存库

2020-04-27 08:31:29

单例模式Python软件设计模式

2021-12-06 08:59:40

RabbitMQ单例写法

2021-06-09 07:01:30

前端CallApply

2015-10-27 09:19:24

2021-09-07 10:44:35

异步单例模式

2020-10-27 12:07:17

DevOps单页应用程序开发

2023-10-18 17:49:58

数据结构队列结构
点赞
收藏

51CTO技术栈公众号