java 单例模式

Java中实现单例模式的常见方式有两种:懒汉式和饿汉式。以下是这两种方式的简单示例:

饿汉式

饿汉式单例模式在类加载时就完成了实例的初始化,以空间换时间,确保了实例的唯一性。

java 复制代码
public class Singleton {

    // 在自己内部定义自己一个实例,是不会被外界访问的

    private static final Singleton instance = new Singleton();

    // 构造器私有化,外部无法直接通过new来实例化对象

    private Singleton() {}

    // 对外提供一个公共的静态方法,返回唯一实例

    public static Singleton getInstance() {

        return instance;

    }

}

懒汉式

懒汉式单例模式只有在真正需要使用时,才会创建实例,实现了懒加载。

java 复制代码
public class Singleton {

    // 声明变量

    private static Singleton instance = null;

    // 构造器私有化

    private Singleton() {}

    // 提供一个静态方法,当使用到该方法时,才去创建instance

    public static synchronized Singleton getInstance() {

        if (instance == null) {

            instance = new Singleton();

        }

        return instance;

    }

}

懒汉式虽然实现了懒加载,但是在多线程环境下可能会创建多个实例,因此需要加锁处理,这就是上面示例中使用synchronized关键字的原因。但是,加锁会影响效率。

java 复制代码
public class Singleton {
    public static void main(String[] args)  {
        Runnable task = () -> {
            Singleton1 instance = Singleton1.getInstance();
            System.out.println(instance);
        };

        for (int i = 0; i < 100; i++) {
            new Thread(task).start();
        }
    }
}

class Singleton1  {
    private static Singleton1 instance;

    private Singleton1() {
        System.out.println("LazySingleton instance created.");
    }

    public  static Singleton1 getInstance() {
        if (instance == null) {
            instance = new Singleton1();
        }
        return instance;
    }
}

双重检查锁定(Double-Checked Locking)

为了解决懒汉式的线程安全问题,同时又不牺牲效率,可以使用双重检查锁定模式。

java 复制代码
public class Singleton {

    // 使用volatile关键字保其可见性和有序性

    private static volatile Singleton instance = null;

    private Singleton() {}

    public static Singleton getInstance() {

        if (instance == null) {

            synchronized (Singleton.class) {

                if (instance == null) {

                    instance = new Singleton();

                }

            }

        }

        return instance;

    }

}

这种方式既实现了线程安全,又避免了每次调用getInstance方法时都需要进行同步。

枚举方式

Java语言规范提供的一种更简洁的单例实现方式是使用枚举。

java 复制代码
public enum Singleton {

    INSTANCE;

    public void someMethod() {

        // 实例方法

    }

}

使用枚举方式实现单例不仅简洁,而且提供了序列化机制,并由JVM从根本上提供保障,避免多次实例化,是实现单例模式的最佳方法。

相关推荐
东北甜妹4 分钟前
Redis Cluster 操作命令
java·开发语言
花间相见6 分钟前
【大模型微调与部署01】—— ms-swift-3.12入门:安装、快速上手
开发语言·ios·swift
techdashen14 分钟前
Rust 正式成立 Types Team:类型系统终于有了专属团队
开发语言·后端·rust
jiayong2317 分钟前
第 17 课:任务选择与批量操作
开发语言·前端·javascript·vue.js·学习
消失的旧时光-194323 分钟前
Spring Boot 核心机制之 @Conditional:从原理到实战(一次讲透)
java·spring boot·后端
量子炒饭大师26 分钟前
【C++11】RAII 义体加装指南 ——【包装器 与 异常】C++11中什么是包装器?有哪些包装器?C++常见异常有哪些?(附带完整代码讲解)
开发语言·c++·c++11·异常·包装器
石榴树下的七彩鱼27 分钟前
智能抠图 API 接入实战:3 行代码实现图片自动去背景(Python / Java / PHP / JS)
java·图像处理·人工智能·python·php·api·抠图
telllong29 分钟前
Python异步编程从入门到不懵:asyncio实战踩坑指南
开发语言·python
知兀30 分钟前
【Result类】(使用/不使用<T> data的情况);自带静态方法、纯数据类;
java·开发语言
达帮主31 分钟前
25.C语言 递归函数
c语言·开发语言·汇编