独一无二的设计模式——单例模式(Java实现)

news/2024/7/7 20:43:45 标签: 后端, 设计模式, 单例模式, java

1. 引言

亲爱的读者们,欢迎来到我们的设计模式专题,今天的讲解的设计模式,还是单例模式哦!上次讲解的单例模式是基于Python实现(独一无二的设计模式——单例模式(python实现))的,但是目前很多实际系统是通过JAVA实现的,所以今天聊一聊基于JAVA的语言特性,单例模式的实现和应用。

2. 什么是单例模式

单例模式(Singleton Pattern)是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点。就像世界上只有一个太阳,我们也希望某些对象在整个应用程序中只有一个实例。单例模式适用于需要全局唯一访问的资源,如数据库连接、配置管理器、日志记录器等。

3. 单例模式的实现(Java)

基本实现

在Java中,单例模式通常使用懒汉式或饿汉式实现,以下讲解四种主要的实现方法:

懒汉式实现,懒汉式单例模式只有在第一次调用getInstance方法时才会创建实例:
java">public class Singleton {
    private static Singleton instance;

    private Singleton() {}

    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}
详细代码解析
  • instance静态变量用于存储单例实例,在类加载时,它被初始化为null
  • getInstance方法是获取实例的关键,如果instance为空,则创建一个新的实例并返回,否则返回已有的实例;
  • 私有构造函数Singleton()防止类被外部实例化,确保实例只能通过getInstance方法获取。
多线程安全的懒汉式实现,为了应对多线程环境,可以使用同步块来确保线程安全:
java">public class Singleton {
    private static Singleton instance;
    private static final Object lock = new Object();

    private Singleton() {}

    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (lock) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}
详细代码解析
  • lock是一个同步对象,用于确保在多线程环境下,只有一个线程能够创建实例;
  • getInstance方法中使用synchronized块来加锁,确保只有一个线程能够进入创建实例的代码块。
饿汉式实现,饿汉式单例在类加载时就创建实例,相比懒汉式避免了多线程问题:
java">public class Singleton {
    private static final Singleton instance = new Singleton();

    private Singleton() {}

    public static Singleton getInstance() {
        return instance;
    }
}
详细代码解析
  • instance静态变量在类加载时即被初始化,确保实例的唯一性和线程安全;
  • 私有构造函数防止外部实例化。
静态内部类实现,这种方法利用类加载机制,只有在调用getInstance时才会创建实例,确保线程安全和延迟加载:
java">public class Singleton {
    private Singleton() {}

    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}
详细代码解析
  • SingletonHolder是一个静态内部类,包含了Singleton的唯一实例;
  • getInstance方法中返回SingletonHolder.INSTANCE,确保实例的延迟加载和线程安全。

4. 单例模式的应用场景和实例

示例一:配置文件管理

在应用程序中,配置文件通常需要全局访问且不应被重复加载。使用单例模式可以确保配置管理器只有一个实例,从而避免重复加载配置文件。

java">import java.util.Properties;

public class ConfigurationManager {
    private static ConfigurationManager instance;
    private Properties config;

    private ConfigurationManager() {
        config = new Properties();
    }

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

    public void setConfig(String key, String value) {
        config.setProperty(key, value);
    }

    public String getConfig(String key) {
        return config.getProperty(key);
    }
}

使用示例:

java">ConfigurationManager configManager = ConfigurationManager.getInstance();
configManager.setConfig("api_url", "https://api.example.com");
System.out.println(configManager.getConfig("api_url"));
示例二:日志记录

日志记录器是单例模式的经典应用之一,通过确保日志记录器的唯一性,我们可以统一管理日志输出,避免多个日志实例之间的混乱:

java">import java.util.logging.*;

public class LoggerSingleton {
    private static LoggerSingleton instance;
    private Logger logger;

    private LoggerSingleton() {
        logger = Logger.getLogger("singleton_logger");
        ConsoleHandler handler = new ConsoleHandler();
        SimpleFormatter formatter = new SimpleFormatter();
        handler.setFormatter(formatter);
        logger.addHandler(handler);
        logger.setLevel(Level.INFO);
    }

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

    public void log(String message) {
        logger.info(message);
    }
}

使用示例:

java">LoggerSingleton logger = LoggerSingleton.getInstance();
logger.log("This is a log message.");

5. 单例模式的优缺点

优点
  • 控制实例数量:确保一个类只有一个实例,节省资源;
  • 全局访问点:提供一个全局访问点,方便管理和使用。
缺点
  • 不易扩展:由于单例模式限制了实例的数量,可能不利于扩展;
  • 隐藏依赖关系单例模式通过全局访问点使用实例,可能导致代码依赖关系不明确,不利于测试。

6. 图示

+----------------+
|   Singleton    |
+----------------+
| - _instance    |
| - _lock        |
+----------------+
| + getInstance()|
+----------------+

7. 总结

单例模式是一种简单而强大的设计模式,确保一个类只有一个实例,并提供全局访问点。在实际开发中,单例模式广泛应用于配置管理、日志记录等场景,通过合理地使用单例模式,我们可以有效管理和优化资源,确保系统的一致性和稳定性。

希望今天的分享能让大家对单例模式有更深入的理解,如果你在项目中也使用了单例模式,欢迎留言分享你的经验和见解!
在这里插入图片描述


http://www.niftyadmin.cn/n/5535331.html

相关文章

基于Hadoop平台的电信客服数据的处理与分析③项目开发:搭建基于Hadoop的全分布式集群---任务7:格式化并启动Hadoop集群

任务描述 任务内容为格式化并启动Hadoop集群,并修复可能出现的Bug。 任务指导 Hadoop集群启动前需要在NameNode上格式化元数据,成功格式化后才能启动Hadoop的HDFS和YARN。 格式化启动Hadoop集群的步骤如下: 1. 在NameNode(ma…

【Python机器学习】算法链与管道——通用的管道接口

Pipeline类补单可以用于预处理和分类,实际上还可以将任意数量的估计器连接在一起。例如,我们可以构建一个包含特征提取、特征选择、缩放和分类的管道,总共有4个步骤。同样的,最后一步可以用聚类或回归代替。 对于管道中估计器的唯…

基于 Gunicorn + Flask + Docker 的模型高并发部署

在现代 Web 应用程序中,处理高并发请求是一个常见且重要的需求。本文将介绍如何使用 Gunicorn、Flask 和 Docker 来实现模型的高并发部署。我们将从环境设置、代码实现、Docker 镜像构建及部署等方面进行详细讲解。 一、环境设置 1. 安装 Flask 首先,…

LLaVA1.5训练数据和时间分析

LLaVA的PT+SFT训练_llava sft-CSDN博客文章浏览阅读379次。这个阶段,使用8个A100(80G)训练LLaVA-v1.5-13B大约需要20h。全量微调,非lora跑不起来啊,以前一直用swift,llama-factory这种框架式的代码库,但用原作者开源的代码也是有很多好处的。在这个阶段,使用 8 个 A100(…

微信小程序 DOM 问题

DOM 渲染问题 问题 Dom limit exceeded, please check if theres any mistake youve made.测试页面 1 <template><scroll-view scroll"screen" style"width: 100%;height: 100vh;" :scroll-y"true" :scroll-with-animation"tru…

vue使用scrollreveal和animejs实现页面滑动到指定位置后再开始执行动画效果

效果图 效果链接&#xff1a;http://website.livequeen.top 介绍 一、Scrollreveal ScrollReveal 是一个 JavaScript 库&#xff0c;用于在元素进入/离开视口时轻松实现动画效果。 ScrollReveal 官网链接&#xff1a;ScrollReveal 二、animejs animejs是一个好用的动画库…

大数据------JavaWeb------MyBatis(完整知识点汇总)

MyBatis MyBatis简介 定义 它是一款优秀的持久层框架&#xff0c;用于简化JDBC开发它原来是Apache的一个开源项目iBatis&#xff0c;后来改名为MyBatis中文官网&#xff1a;https://mybatis.org/mybatis-3/zh_CN/index.html JaveEE三层架构 表现层&#xff08;做页面展示&…

什么是代理IP?代理IP有什么好处呢?

在互联网的世界里&#xff0c;每台设备都有一个独特的身份标识&#xff0c;即IP地址。这个IP地址不仅仅是一串数字&#xff0c;它代表着设备在网络中的位置&#xff0c;同时也可能暴露用户的地理位置和其他个人信息。为了保护用户的隐私&#xff0c;提高网络安全&#xff0c;或…