Java设计模式之单例模式

简介: 在软件工程中,单例模式是一种常用的设计模式,其核心目标是确保一个类只有一个实例,并提供一个全局访问点来获取这个实例。Java作为一门广泛使用的编程语言,实现单例模式是面试和实际开发中的常见需求。

在软件工程中,单例模式是一种常用的设计模式,其核心目标是确保一个类只有一个实例,并提供一个全局访问点来获取这个实例。Java作为一门广泛使用的编程语言,实现单例模式是面试和实际开发中的常见需求。本文将深入探讨Java中的单例模式,包括其优缺点分析、实现方式等。

singleton.jpg

优缺点分析

优点

  • 内存利用率高:因为只创建一个实例,减少了内存的开销。
  • 性能提高:避免了频繁的创建和销毁对象所带来的性能开销。
  • 共享资源:可以方便地共享数据。

缺点

  • 不支持有参数的构造函数:单例模式通常不能接受参数,这限制了其使用场景。
  • 扩展困难:由于单例模式的特殊性,扩展时需要修改代码,违背了“开闭原则”。
  • 隐藏类之间的依赖关系:由于直接使用了单例类,使得类之间的依赖关系不明显。

单例模式的实现

单例模式的实现比较简单,每次获取实例之前先判断其是否存在,不存在则创建,存在则直接返回。单例模式的实例只能由其自身去创建和销毁,不允许其它类通过new关键字去创建。单例模式常见的写法由饿汉式、懒汉式(线程安全)、双重检查锁、静态内部类。以下是这些写法的具体实现。

饿汉式

饿汉式是指在类加载时就完成了实例的初始化,本质上是利用类加载机制保证实例的唯一性和线程安全。

package cn.xj.xjdoc.singleton;

/**
 * 单例模式--饿汉式
 */
public class HungrySingleton {
   
   
    private static final HungrySingleton INSTANCE = new HungrySingleton();

    /**
     * 构造方法私有化
     * 保证外部无法直接通过new关键字来创建此类的实例
     */
    private HungrySingleton(){
   
   

    }

    /**
     *获取实例的public的静态方法
     */
    public static HungrySingleton getInstance(){
   
   
        return INSTANCE;
    }
}

懒汉式

懒汉式指全局的单例实例在第一次被使用时构建。为了解决线程安全问题,增加synchronized关键字,使得在同一时刻,只有一个线程能够访问这个方法。

package cn.xj.xjdoc.singleton;

/**
 * 单例模式--懒汉式
 */
public class LazySingleton {
   
   

    private static LazySingleton instance;

    /**
     * 构造方法私有化
     * 保证外部无法直接通过new关键字来创建此类的实例
     */
    private LazySingleton(){
   
   

    }

    /**
     *获取实例的public的静态方法,
     *使用synchronized锁,保证在同一时刻,只有一个线程能执行该方法,避免了多线程下创建多个实例的问题。
     */
    public static synchronized LazySingleton getInstance(){
   
   
        //实例为null则创建实例
        if(instance == null){
   
   
            instance = new LazySingleton();
        }

        return instance;
    }

}

双重检查锁

双重检查锁定在懒汉式的基础上做了进一步的优化,既保证了懒加载,又保证了线程安全,同时减少了锁的粒度,提高了性能。

package cn.xj.xjdoc.singleton;

/**
 * 单例模式--双重检查锁
 */
public class DoubleCheckedLockSingleton {
   
   

    // 声明一个私有的静态变量instance,并使用volatile关键字修饰。
    // volatile确保instance变量的可见性和禁止指令重排序,对于双重检查锁定模式非常关键。
    private volatile static DoubleCheckedLockSingleton instance;

    /**
     * 构造方法私有化
     * 保证外部无法直接通过new关键字来创建此类的实例
     */
    private DoubleCheckedLockSingleton(){
   
   

    }

    /**
     *获取实例的public的静态方法
     */
    public static DoubleCheckedLockSingleton getInstance(){
   
   
        // 第一次检查:如果instance不为null,则不需要进入同步块,直接返回实例。
        // 这样做的好处是,只有第一次调用时才会进行同步,之后的调用由于instance已经被实例化,直接返回,提高了效率。
        if(instance == null){
   
   
            // 使用synchronized锁,,确保多线程环境下的线程安全
            synchronized (DoubleCheckedLockSingleton.class){
   
   
                //实例为null则创建实例
                if(instance == null){
   
   
                    instance = new DoubleCheckedLockSingleton();
                }
            }
        }
        return instance;
    }

}

静态内部类

静态内部类单例模式的实现思路是:利用JVM在加载外部类的过程中不会加载静态内部类,只有在内部类被使用时才会被加载。这样不仅能延迟内部类的加载时间,还能保证线程安全。

package cn.xj.xjdoc.singleton;


/**
 * 单例模式--静态内部类
 */
public class StaticNestedClassSingleton {
   
   
    /**
     * 构造方法私有化
     * 保证外部无法直接通过new关键字来创建此类的实例
     */
    private StaticNestedClassSingleton(){
   
   

    }

    // 静态内部类
    private static class SingletonHolder {
   
   
        // 在内部类中持有外部类的实例,并且可以被直接初始化
        private static final StaticNestedClassSingleton INSTANCE = new StaticNestedClassSingleton();
    }
    /**
     *获取实例的public的静态方法
     */
    public static StaticNestedClassSingleton getInstance(){
   
   
        return SingletonHolder.INSTANCE;
    }
}

总结

单例模式是一种简单但非常实用的设计模式,在Java开发中有广泛的应用。选择合适的实现方式,可以在保证性能的同时,确保程序运行的正确性。然而,任何设计模式都不是万能的,使用单例模式时也应该考虑其适用场景和潜在的缺点。

目录
相关文章
|
12天前
|
设计模式 安全 Java
Java编程中的单例模式:理解与实践
【10月更文挑战第31天】在Java的世界里,单例模式是一种优雅的解决方案,它确保一个类只有一个实例,并提供一个全局访问点。本文将深入探讨单例模式的实现方式、使用场景及其优缺点,同时提供代码示例以加深理解。无论你是Java新手还是有经验的开发者,掌握单例模式都将是你技能库中的宝贵财富。
18 2
|
30天前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
27 2
|
22天前
|
设计模式 安全 Java
Java编程中的单例模式深入剖析
【10月更文挑战第21天】在Java的世界里,单例模式是设计模式中一个常见而又强大的存在。它确保了一个类只有一个实例,并提供一个全局访问点。本文将深入探讨如何正确实现单例模式,包括常见的实现方式、优缺点分析以及最佳实践,同时也会通过实际代码示例来加深理解。无论你是Java新手还是资深开发者,这篇文章都将为你提供宝贵的见解和技巧。
92 65
|
11天前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
12天前
|
设计模式 安全 Java
Java编程中的单例模式深入解析
【10月更文挑战第31天】在编程世界中,设计模式就像是建筑中的蓝图,它们定义了解决常见问题的最佳实践。本文将通过浅显易懂的语言带你深入了解Java中广泛应用的单例模式,并展示如何实现它。
|
19天前
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入理解与应用
【10月更文挑战第22天】 在软件开发中,设计模式是解决特定问题的通用解决方案。本文将通过通俗易懂的语言和实例,深入探讨PHP中单例模式的概念、实现方法及其在实际开发中的应用,帮助读者更好地理解和运用这一重要的设计模式。
14 1
|
23天前
|
设计模式 Java 程序员
[Java]23种设计模式
本文介绍了设计模式的概念及其七大原则,强调了设计模式在提高代码重用性、可读性、可扩展性和可靠性方面的作用。文章还简要概述了23种设计模式,并提供了进一步学习的资源链接。
36 0
[Java]23种设计模式
|
7天前
|
设计模式 JavaScript Java
Java设计模式:建造者模式详解
建造者模式是一种创建型设计模式,通过将复杂对象的构建过程与表示分离,使得相同的构建过程可以创建不同的表示。本文详细介绍了建造者模式的原理、背景、应用场景及实际Demo,帮助读者更好地理解和应用这一模式。
|
1月前
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入解析与实践
在PHP开发中,设计模式是提高代码可维护性、扩展性和复用性的关键技术之一。本文将通过探讨单例模式,一种最常用的设计模式,来揭示其在PHP中的应用及优势。单例模式确保一个类仅有一个实例,并提供一个全局访问点。通过实际案例,我们将展示如何在PHP项目中有效实现单例模式,以及如何利用这一模式优化资源配置和管理。无论是PHP初学者还是经验丰富的开发者,都能从本文中获得有价值的见解和技巧,进而提升自己的编程实践。
|
1月前
|
设计模式 安全 Java
C# 一分钟浅谈:设计模式之单例模式
【10月更文挑战第9天】单例模式是软件开发中最常用的设计模式之一,旨在确保一个类只有一个实例,并提供一个全局访问点。本文介绍了单例模式的基本概念、实现方式(包括饿汉式、懒汉式和使用 `Lazy<T>` 的方法)、常见问题(如多线程和序列化问题)及其解决方案,并通过代码示例详细说明了这些内容。希望本文能帮助你在实际开发中更好地应用单例模式,提高代码质量和可维护性。
29 1

热门文章

最新文章

  • 1
    C++一分钟之-设计模式:工厂模式与抽象工厂
    43
  • 2
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    46
  • 3
    C++一分钟之-C++中的设计模式:单例模式
    54
  • 4
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    38
  • 5
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    62
  • 6
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    57
  • 7
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    41
  • 8
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    50
  • 9
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    106
  • 10
    Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
    78