首页 > 其他分享 >单例模式及其思想

单例模式及其思想

时间:2024-08-01 22:28:07浏览次数:5  
标签:Singleton 思想 模式 实例 static private 单例 public

本文包括以下几点↓

在这里插入图片描述

结论设计模式不是简单地将一个固定的代码框架套用到项目中,而是一种严谨的编程思想,旨在提供解决特定问题的经验和指导。

单例模式(Singleton Pattern)

意图

旨在确保类只有一个实例,并提供一个全局访问点以访问该实例。

适用性

当你希望系统中只有一个实例,并且需要从全局任何地方都能访问该实例时。

当你需要严格控制某个类的实例化过程,以确保所有代码都使用相同的实例时。

简述

单例模式的核心思想是控制对象只实例化一次,并提供一个全局访问点来获取该实例。这样可以确保在系统中只有一个实例,并且所有对该实例的访问都通过同一个访问点。

实现单例模式有几种方式,通过以下案例展开。

简单案例

六种常见的单例模式实现方式,逐一分析它们的特点和适用场景:

  1. 懒汉式之线程不安全:

源码

public class Singleton {  
    private static Singleton instance;  
    private Singleton (){}  

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

特点: 懒加载,在首次使用时创建实例化对象。在多线程环境下,可能会出现并发问题,导致创建多个实例。

适用场景: 单线程环境下的简单应用,对性能要求不高,且不会频繁使用单例的场景。

  1. 懒汉式之线程安全

源码

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

特点: 同样是懒加载,但使用了 synchronized 关键字来保证线程安全。然而,这会造成性能下降,因为每次获取实例对象都需要加锁。

适用场景: 在多线程环境下使用,对性能要求不高的场景,或者对并发要求不是特别高的情况。

  1. 饿汉式

源码

public class Singleton {  
    private static Singleton instance = new Singleton();  
    private Singleton (){}  
    public static Singleton getInstance() {  
        return instance;  
    }  
}

特点: 在类加载时就创建实例化对象,因此不存在多线程安全问题,且访问时效率高。但可能会浪费内存,因为实例在程序启动时就创建了,即使未被使用也会占用内存。

适用场景: 对内存占用没有过多要求,且希望在程序启动时就初始化单例的场景。

  1. 双检锁/双重校验锁

源码

public class Singleton {  
    private volatile static Singleton singleton;
    private Singleton (){}
    public static Singleton getSingleton() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

特点: 通过双重检查加锁,既实现了懒加载,又保证了线程安全,且在实例已创建的情况下避免了不必要的锁竞争,因此性能较高。

适用场景: 对性能要求较高,且需要懒加载的场景,特别是在多线程环境下。

  1. 登记式/静态内部类

源码

public class Singleton {
    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }
    private Singleton (){}
    public static final Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

特点: 通过静态内部类实现了懒加载,且保证了线程安全。在类加载时并不会初始化,只有在首次调用 getInstance() 方法时才会初始化。

适用场景: 对内存占用有要求,且希望在首次使用时才初始化的场景。

  1. 枚举类

源码

public enum Singleton {
    INSTANCE;
//  private Singleton(){}
    public Singleton getSelf() {
        return INSTANCE;
    }
}
public static void main(String[] args) {
    Singleton instance = Singleton.INSTANCE;
}

特点: 使用枚举类型实现单例,利用枚举的特性保证了在任何情况下都只会有一个实例存在。枚举类的实例在类加载时就会被初始化。

适用场景: 对内存占用没有过多要求,且希望简洁明了地实现单例的场景。枚举类也是线程安全的。

枚举类本身就是单例模式Singleton instance = Singleton.INSTANCE; 并且可以有多个枚举INSTANCE,TWO;枚举类中的INSTANCE就是Singleton对象实例,看getSelf方法不报错就知道了;枚举类的构造方法本身就是private,放开实例中的注释结果一样;枚举中的方法只能通过Singleton.INSTANCE.getSelf()的方式来调用,这就是单例的表现。

这是实现单例模式的最佳方法,它更简洁,自动支持序列化机制,绝对防止多次实例化。但不能通过反射来调用私有构造方法。

在Spring框架中的应用??

在Spring框架中,使用到单例模式思想的地方非常多,被广泛应用于管理和创建bean实例。Spring容器默认情况下会将所有的bean都配置为单例,这意味着在整个应用程序中,每个bean都只有一个实例存在。这种默认的单例模式确保了Spring应用程序的性能和资源利用率。

在Spring上下文中,通过使用@Bean@Component@Service@Repository 等注解或在配置文件中显式声明bean,Spring会自动将其配置为单例模式,即在整个应用程序中只创建一个实例。比如配置类的创建。

例如:数据库连接配置类

@Configuration
public class AppConfig {

    @Bean
    public DataSource dataSource() {
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/mydatabase");
        dataSource.setUsername("username");
        dataSource.setPassword("password");
        return dataSource;
    }
}

在Spring框架中的这个单例模式是使用单例设计模式创建Bean的吗?

很显然不是。最显著的一点就是,你在项目中创建的类的构造方法是私有的吗?

Spring 底层创建单例模式的Bean 使用的是BeanFactory工厂设计模式去创建的,在此别混淆了。这种Bean概念更准确的描述应该是Bean的单例作用域

除此之外,Spring框架还提供了其他类型的作用域,比如Prototype原型模式,可以通过 @Scope("prototype") 注解来标注一个bean,这样每次从容器中获取该bean时,容器都会创建一个新的实例。

@Component
@Scope("prototype")
public class MyPrototypeBean {
    // 类的具体代码
}

Prototype原型作用域的Bean对象和原型设计模式有什么关系呢?在原型设计模式中再进行分享。

总结:

单例模式的作用主要包括以下几个方面:

  1. 确保全局唯一实例: 单例模式确保系统中某个类只有一个实例存在,无论在何处进行实例化,都会获得相同的实例。这可以防止多个对象之间的状态不一致或冲突,确保所有使用者操作一致,有助于管理全局资源。
  2. 提供全局访问点: 单例模式提供一个全局访问点,使得任何地方都可以轻松访问该实例。这在需要频繁访问某个对象或共享资源的情况下非常有用,可以简化代码逻辑,提高代码的可读性和可维护性。
  3. 节约系统资源: 由于单例模式只创建一个对象实例,因此可以避免重复创建对象所带来的性能和内存消耗。特别是在频繁访问该对象的场景下,单例模式可以有效减少系统资源的使用。
  4. 控制对象的实例化过程: 单例模式可以对对象的实例化过程进行严格控制,确保只有一个实例存在,并且在运行时动态创建或延迟实例化,以满足特定的需求。这在需要对对象的创建过程进行特殊处理或优化时非常有用。
  5. 避免全局变量的滥用: 单例模式可以将全局状态封装在一个对象中,避免使用大量的全局变量和静态方法,从而提高代码的可维护性和可测试性。

简言之,单例模式在需要确保系统中某个类只有一个实例存在,并且需要提供全局访问点。它可以简化系统架构,提高代码的重用性和可维护性,同时节约系统资源,提高系统性能。但是要注意,线程是否安全的问题。

看了Spring单例作用域的Bean,应该知道一点:

设计模式不是简单地将一个固定的代码框架套用到项目中,而是一种严谨的编程思想,旨在提供解决特定问题的经验和指导。

就单例设计模式来说,控制一个类只有一个实例,并提供一个全局访问点,掌握这个编程思想的应用才是学习设计模式的关键。

在这里插入图片描述

软考中级--软件设计师毫无保留的备考分享

2023年下半年软考考试重磅消息

通过软考后却领取不到实体证书?

计算机算法设计与分析(第5版)

Java全栈学习路线、学习资源和面试题一条龙

软考证书=职称证书?

什么是设计模式?

标签:Singleton,思想,模式,实例,static,private,单例,public
From: https://www.cnblogs.com/dennyLee2025/p/18337717

相关文章

  • ObjectARX 判断实体是否是在位编辑块对象简单例子
    判断使用acdbIsInLongTransaction应该就可以。ads_nameent;ads_pointpt;if(RTNORM!=acedEntSel(_T("\n选择对象:"),ent,pt)){return;}AcDbObjectIdobjId;acdbGetObjectId(objId,ent);//直接判断//if(acdbIsInLongTransaction......
  • 冒泡排序的具体思想和算法实现以及改进
    冒泡排序——稳定算法从小到大排序:0~length-1对比a[0]和a[1],如果前一个大于后一个,交换位置。对比a[1]和a[2],如果前一个大于后一个,交换位置。对比a[2]和a[3],如果前一个大于后一个,交换位置。...对比a[length-2]和a[length-1],如果前一个大于后一个,交换位置。第一轮结果下......
  • 折半插入排序算法思想及代码实现
    折半插入排序(BinaryInsertionSort)是插入排序算法的一种优化版本。插入排序的基本思想是将一个记录插入到已经排序好的有序表中,从而得到一个新的、记录数增加1的有序表。传统的插入排序在寻找插入位置时,采用的是顺序比较的方式,即逐个与有序表中的元素进行比较,直到找到比待插入......
  • 从“人巡”到“智控”:EasyCVR智能视频监控技术变革河道违建监测模式
    一、背景分析随着城市化进程的加快,河道作为城市生态系统的重要组成部分,其保护与管理日益受到重视。然而,非法侵占河道、违规建设等行为时有发生,不仅破坏了河道的自然生态,还严重威胁到防洪安全和水质安全。为了有效解决这一问题,EasyCVR智能视频监控技术凭借其高效、精准、全天候的......
  • 策略模式
    if条件的终结者,不同的条件调用不同的处理类进行处理;1.定义一个顶层接口,继承InitializingBean接口,保证在容器启动时将实现该接口的类实例化交给容器管理;publicinterfaceImportDataHandler<T>extendsInitializingBean{/***处理导入数据**@paramb......
  • 模板方法模式
    上层抽象类定义好操作的基本框架,一些特殊的子操作交给子类去实现,使得子类可以在不改变上层基类的情况下,可以定制操作的某一步骤。抽象类:模版方法:定义操作的骨架基本方法抽象方法:交给子类实现具体方法:基类自己实现,子类也可以进行覆盖(重写)具体实现类实现......
  • 单例模式
    在整个系统中,只能存在一个对象实例,也就是说只能new一次饿汉式:在系统启动后,就会实例化一个对象(不管用不用得到,我先买了再说)缺点:造成内存浪费(如果从来没有使用过)优点:不会造成线程安全的问题classEHPatternInstance1{privatefinalstaticEHPatternInstance1IN......
  • 适配器模式
    适配器模式(AdapterPattern)将某个类的接口转换成客户端期望的另一个接口表示,主要目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作适配器继承适配对象,并实现适配目标所有的方法 国标接口(适配目标)publicinterfaceNationalStandard{voiduse()......
  • Calico VXLAN 模式
    CalicoVxLAN模式一、环境信息主机IPubuntu172.16.94.141软件版本docker26.1.4helmv3.15.0-rc.2kind0.18.0clab0.54.2kubernetes1.23.4ubuntuosUbuntu20.04.6LTSkernel5.11.5内核升级文档二、安装服务kind配置文件......
  • Zabbix监控模式及proxy代理
    一、zabbix主动与被动监控模式被动模式和主动模式都是相对于agent客户端而言的主动模式:客户端主动将数据发给server或者proxy被动模式:server或者proxy主动去找客户端索要数据1.被动模式默认是被动模式被动模式是说zabbixagent被动的接受zabbixserver周期性发送......