手撕单例的 5 种写法!

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

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

alt

具体实现如下。

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 种:饿汉模式、懒汉非安全模式、懒汉安全模式、双重效验锁模式、静态内部类模式,但它的写法基本都是以下三步:

  1. 定义私有构造方法(防止 new 多个实例)。
  2. 定义私有变量(承接单例对象)。
  3. 定义统一返回对象的方法。
#java#
全部评论

相关推荐

点赞 评论 收藏
分享
使用AJAX进行异步通信的基本步骤如下:https://www.nowcoder.com/issue/tutorial?zhuanlanId=Mg58Em&uuid=43521d43a8e341f888324dd690363024创建XMLHttpRequest对象:使用JavaScript代码创建一个XMLHttpRequest对象,该对象用于进行异步通信。为XMLHttpRequest对象添加事件监听器:为XMLHttpRequest对象添加事件监听器,以便在通信状态改变时接收回调。创建请求:使用XMLHttpRequest对象的open方法创建一个HTTP请求。其中,需要指定请求的方法(GET或POST)和目标URL。设置请求头部:使用XMLHttpRequest对象的setRequestHeader方法设置请求头部,以便向服务器传递必要的信息,如数据格式等。发送请求:使用XMLHttpRequest对象的send方法发送请求。对于GET请求,可以将参数拼接到URL后面;对于POST请求,可以将参数以字符串或FormData对象的形式传递。处理响应:在XMLHttpRequest对象的事件回调函数中,使用responseText或responseXML属性来获取服务器的响应数据。可以使用这些数据来更新页面或进行其他操作。AJAX的原理是通过XMLHttpRequest对象实现与服务器的异步通信。在传统的同步通信中,浏览器发起请求后需要等待服务器响应,并在等待期间无法进行其他操作。而使用AJAX进行异步通信时,浏览器可以在发送请求后继续执行其他代码,不需要等待服务器响应。当响应返回后,浏览器会调用注册的回调函数来处理响应数据,从而实现异步更新页面内容。AJAX主要用于以下方面:https://www.nowcoder.com/issue/tutorial?zhuanlanId=Mg58Em&uuid=43521d43a8e341f888324dd690363024动态加载内容:可以在页面加载完成后通过AJAX请求服务器获取额外的内容,如文章列表、评论等。表单数据提交:可以通过AJAX将表单数据异步提交给服务器,而不需要刷新整个页面。轮询更新:可以周期性地向服务器发送请求,以获取热点数据的更新。需要注意的是,由于AJAX请求涉及跨域问题,可能会遇到安全性限制。在跨域请求时,需要服务器允许相关的请求,并且需要特别处理响应的数据。
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务