热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

深入理解GoF设计模式之单例模式

本文详细介绍了单例模式,这是一种确保类的实例在整个应用程序生命周期中始终保持唯一的创建型设计模式。文章不仅探讨了单例模式的基本概念和优势,还分析了其潜在的缺点,如违反单一职责原则。此外,文中提供了多种实现单例模式的方法,包括饿汉式、懒汉式、基于DCL的双重检查锁、静态内部类以及基于枚举的实现。

单例模式是一种广泛使用的创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将详细介绍单例模式的概念、实现方式及其优缺点。

概述

单例模式的核心目标是确保一个类的实例在整个应用程序中始终唯一。为了实现这一点,通常需要将类的构造函数设为私有,并在类内部创建实例,然后通过一个公共的静态方法提供对该实例的访问。这种方式可以有效避免因频繁创建和销毁实例而导致的性能开销,但也存在一些缺点,比如违反了单一职责原则,即一个类应该只有一个引起变化的原因。

实现方式

1. 饿汉式

饿汉式单例是最简单的实现方式之一。在这种模式下,类的实例在类加载时就已经创建好了,因此是线程安全的。然而,这种方式可能导致资源的浪费,因为即使实例从未被使用,也会占用内存。

public class SingletonDemo1 {
private static final SingletonDemo1 instance = new SingletonDemo1("我是饿汉式的单例");
private String description;

private SingletonDemo1(String description) {
this.description = description;
}

public void getInfo() {
System.out.println(description);
}

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

测试结果如下所示:

图1

2. 懒汉式

懒汉式单例实现了延迟加载,即在首次需要实例时才创建。这种方式可以节省内存,但默认情况下不是线程安全的。可以通过同步方法来解决线程安全问题,但这会影响性能。

public class SingletonDemo2 {
private static SingletonDemo2 instance = null;
private String description;

private SingletonDemo2(String description) {
this.description = description;
}

public void getInfo() {
System.out.println(description);
}

public static synchronized SingletonDemo2 getInstance() {
if (instance == null) {
instance = new SingletonDemo2("我是线程安全的懒汉式单例");
}
return instance;
}
}

测试结果如下所示:

图2

3. 基于DCL的双重检查锁

基于DCL的双重检查锁是一种既实现了延迟加载又保证了线程安全的单例实现方式。它通过两次检查实例是否为空来减少不必要的同步操作,提高性能。在实现中,静态变量instance必须使用volatile关键字修饰,以防止指令重排序问题。

public class SingletonDemo4 {
private static volatile SingletonDemo4 instance = null;
private String description;

private SingletonDemo4(String description) {
this.description = description;
}

public void getInfo() {
System.out.println(description);
}

public static SingletonDemo4 getInstance() {
if (instance == null) {
synchronized (SingletonDemo4.class) {
if (instance == null) {
instance = new SingletonDemo4("我是基于DCL的线程安全的单例");
}
}
}
return instance;
}
}

测试结果如下:

图4

4. 基于静态内部类的单例

基于静态内部类的单例实现方式既实现了延迟加载,又是线程安全的。这种方式利用了Java类加载机制,确保实例在首次调用getInstance方法时才被创建。

public class SingletonDemo5 {
private String description;

private SingletonDemo5(String description) {
this.description = description;
}

public void getInfo() {
System.out.println(description);
}

private static class SingletonDemo5Holder {
private static final SingletonDemo5 instance = new SingletonDemo5("我是基于静态内部类的线程安全的单例");
}

public static SingletonDemo5 getInstance() {
return SingletonDemo5Holder.instance;
}
}

测试结果如下所示:

图5

5. 基于枚举的单例

基于枚举的单例实现方式简洁且线程安全。枚举的构造器默认是私有的,这确保了实例的唯一性。这种方式适用于不需要延迟加载的场景。

public enum SingletonDemo6 {
INSTANCE("我是枚举法的单例");
private String description;

SingletonDemo6(String description) {
this.description = description;
}

public void getInfo() {
System.out.println(description);
}
}

测试结果如下:

图6

参考文献
  1. Head First 设计模式 弗里曼著

推荐阅读
  • 本文总结了Java程序设计第一周的学习内容,涵盖语言基础、编译解释过程及基本数据类型等核心知识点。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
  • 本文介绍如何在Java项目中使用Log4j库进行日志记录。我们将详细说明Log4j库的引入、配置及简单应用,帮助开发者快速上手。 ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
author-avatar
从容面对天下事
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有