设计模式-单例模式实战

目录

一、引言

单例模式是一种创建型设计模式,用于确保一个类只有一个实例,且提供全局访问点以访问该实例。以下是单例模式的关键特点:

  • 私有构造函数:单例类通常会将其构造函数声明为私有,以防止外部直接创建实例。
  • 私有静态实例:单例类通常会维护一个私有的静态实例变量,用于保存唯一的实例。
  • 静态获取方法:单例类通常提供一个静态的获取方法,允许客户端代码访问单例实例。
  • 惰性初始化:单例实例通常是在首次访问时初始化的,以延迟实例化,提高性能。
  • 线程安全:单例模式通常需要考虑线程安全性,以防止多个线程同时创建实例。

二、适用场景

单例模式适用于以下情况:

  1. 全局资源访问:当一个对象需要在系统中全局唯一访问时,如配置管理、日志记录、数据库连接等。
  2. 控制资源访问:当需要限制某种资源(如线程池、连接池、缓存)的访问数量时。
  3. 惰性初始化:当对象的创建和初始化可能具有较高的开销,但只在需要时才进行时。
  4. 线程池:用于创建线程池,以便在整个应用程序中维护一个可管理的线程池实例。

三、代码实战

饿汉式单例模式

java 复制代码
public class EagerSingleton {
    private static final EagerSingleton instance = new EagerSingleton();

    private EagerSingleton() {
        // 私有构造函数
    }

    public static EagerSingleton getInstance() {
        return instance;
    }
}

原理:在类加载时,即实例化前,实例变量就已经被分配内存,因此保证了线程安全。但可能会浪费内存,因为不一定会使用该实例。

懒汉式单例模式

java 复制代码
public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {
        // 私有构造函数
    }

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

原理:该实现使用惰性初始化,只有在第一次调用 getInstance() 时才创建实例。通过加锁来确保线程安全,但可能引发性能问题。

双重检查锁定单例模式

java 复制代码
public class DoubleCheckedSingleton {
    private volatile static DoubleCheckedSingleton instance;

    private DoubleCheckedSingleton() {
        // 私有构造函数
    }

    public static DoubleCheckedSingleton getInstance() {
        if (instance == null) {
            synchronized (DoubleCheckedSingleton.class) {
                if (instance == null) {
                    instance = new DoubleCheckedSingleton();
                }
            }
        }
        return instance;
    }
}

原理:通过双重检查,确保只在第一次创建实例时使用同步锁,提高了性能。

静态内部类单例模式

java 复制代码
public class InnerStaticSingleton {
    private InnerStaticSingleton() {
        // 私有构造函数
    }

    private static class SingletonHolder {
        private static final InnerStaticSingleton instance = new InnerStaticSingleton();
    }

    public static InnerStaticSingleton getInstance() {
        return SingletonHolder.instance;
    }
}

原理:利用Java类加载机制,保证只有在首次访问 getInstance() 时才加载内部类并创建实例。

四、实际应用举例

Java开发工具包(JDK)中有一些类和模块使用了单例模式来管理资源或确保全局唯一性。以下是一些示例:

  1. Runtime类:java.lang.Runtime类是Java的运行时系统入口,它使用了饿汉式单例模式来确保在应用程序中只有一个运行时实例。你可以通过Runtime.getRuntime()方法获取运行时对象的唯一实例,从而执行与运行时环境相关的操作。
  2. Calendar类:java.util.Calendar类是用于日期和时间操作的抽象类,它提供了一个getInstance()方法,返回特定日历系统的单例实例。这是一个懒汉式单例模式的示例,以确保只有一个日历实例。
  3. Desktop类:java.awt.Desktop类用于与桌面操作系统进行交互,如打开文件、浏览网页等。它使用了饿汉式单例模式来保证只有一个桌面对象的实例。

这些示例中的单例模式确保了全局唯一性,并用于管理特定资源或提供全局访问点。虽然不是所有JDK中的类都使用单例模式,但一些核心类确实使用了这种设计模式来确保一些特定的实例只存在一个。

Runtime解析

在 Java 的 Runtime 类中,使用了饿汉式单例模式来确保在应用程序中只有一个运行时实例。以下是 Runtime 类的部分代码,展示了如何使用单例模式来创建和返回运行时实例:

java 复制代码
public class Runtime {
   private static Runtime currentRuntime = new Runtime();

   // 私有构造函数,防止外部实例化
   private Runtime() {
   }

  // 获取当前运行时实例
   public static Runtime getRuntime() {
       return currentRuntime;
   }

   // ...其他方法和功能...
}

解释:

  1. currentRuntime 是一个静态私有成员变量,它在类加载时被初始化。这就是饿汉式单例模式的特点,实例在类加载时就被创建,而不是在首次访问时才创建。
  2. 构造函数 private Runtime() 被声明为私有,这意味着外部代码无法直接实例化 Runtime 类。
  3. getRuntime() 方法是一个静态方法,用于获取当前运行时实例。它返回的是 currentRuntime 静态实例变量,确保每次调用 getRuntime() 方法都返回同一个实例。

这种设计确保了在整个应用程序中只有一个运行时实例,因为 currentRuntime 在类加载时就被创建,并且构造函数是私有的,外部代码无法创建新的实例。这是一种有效的方式,以确保只有一个全局运行时环境,以便执行与运行时环境相关的操作,如执行外部进程、获取系统信息等。

五、结论

以下是单例模式的一些总结和结论:
优点:

  1. 全局唯一性:单例模式确保一个类只有一个实例,因此能够全局唯一地管理特定类型的资源或状态。
  2. 惰性初始化:单例实例通常是在首次访问时初始化的,从而避免了不必要的资源浪费。
  3. 全局访问点:单例提供了一个全局访问点,允许应用程序中的其他对象轻松访问单例实例。
  4. 线程安全性:合理实现的单例模式可以确保多线程环境下的线程安全性,防止多次实例化。

缺点和注意事项:

  1. 过度使用:过度使用单例模式可能导致全局状态的滥用,使代码难以维护和测试。应谨慎使用。
  2. 不适用于所有情况:并非所有类都适合作为单例,有些类需要多个实例来独立处理不同任务。
  3. 测试困难:单例模式的全局状态共享可能会使单元测试变得复杂,因此需要额外的努力来测试应用。
  4. 多例模式:在某些情况下,可能需要多个实例,而不仅仅是一个全局唯一实例。

总的来说,单例模式是一种非常有用的设计模式,用于确保一个类只有一个实例。它在管理全局资源、限制实例数量以及提供全局访问点等方面发挥了重要作用。然而,在使用单例模式时,应慎重考虑,确保它真正符合应用程序的需求。此外,要注意线程安全性和测试问题,以确保单例实现的正确性。

点赞收藏,富婆包养✋✋

相关推荐
白榆maple23 分钟前
(蓝桥杯C/C++)——基础算法(下)
算法
JSU_曾是此间年少28 分钟前
数据结构——线性表与链表
数据结构·c++·算法
许野平41 分钟前
Rust: 利用 chrono 库实现日期和字符串互相转换
开发语言·后端·rust·字符串·转换·日期·chrono
此生只爱蛋1 小时前
【手撕排序2】快速排序
c语言·c++·算法·排序算法
暗黑起源喵2 小时前
设计模式-工厂设计模式
java·开发语言·设计模式
咕咕吖2 小时前
对称二叉树(力扣101)
算法·leetcode·职场和发展
齐 飞2 小时前
MongoDB笔记01-概念与安装
前端·数据库·笔记·后端·mongodb
九圣残炎2 小时前
【从零开始的LeetCode-算法】1456. 定长子串中元音的最大数目
java·算法·leetcode
lulu_gh_yu3 小时前
数据结构之排序补充
c语言·开发语言·数据结构·c++·学习·算法·排序算法
LunarCod3 小时前
WorkFlow源码剖析——Communicator之TCPServer(中)
后端·workflow·c/c++·网络框架·源码剖析·高性能高并发