详解单例模式、模板方法及项目和源码应用

大家好,我是此林。

设计模式为解决特定问题提供了标准化的方法。在项目中合理应用设计模式,可以避免重复解决相同类型的问题,使我们能够更加专注于具体的业务逻辑,减少重复劳动。设计模式在定义系统结构时通常考虑到未来的扩展。例如,工厂模式、策略模式等能让系统在增加新功能时无需改动现有代码,只需扩展新模块即可,减少了修改现有代码的风险。

今天分享的是单例模式和模板模式,这两种设计模式在项目和源码中的使用。

1. 单例模式

一般开发中,我们使用的是 Spring 框架,默认情况下,我们通过 @Bean@Component 等注解注入的 Bean 对象是 单例的(即 Singleton),也就是说 Spring 会在容器启动时创建一个该类型的 Bean 实例,并在整个应用程序上下文中共享这个实例。可以通过 @Scope 注解来指定 Bean 的作用域,控制其生命周期和作用范围。默认的作用域是 @Scope("singleton")。

那 Spring 是如何实现单例模式的呢?

关注源码,发现 Spring 维护了一个全局的单例池(ConcurrentHashMap),key 是 BeanName,value 是 Bean 对象。
DefaultSingletonBeanRegistry 类实现了 SingletonBeanRegistry 接口)

我们在开发过程中使用Bean对象,会根据 BeanName 去单例池中获取 Bean 对象,保证了对象的全局的唯一性。

当然,它和我们平时所说的几种单例模式实现还是不一样的。

1. 饿汉式

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

关键点:

1. 使用 private 关键字,代表外部无法对变量 instance 直接修改

2. 使用 static 关键字,代表 instance 变量在类加载的时候就会被初始化,这个和 JVM 加载类有关。

3. final 关键字的作用

  • final 用于修饰变量时,表示该变量 一旦赋值就不能再被修改。也就是说,变量 引用 一旦指向某个对象,就不能再指向其他对象。
  • final 修饰一个引用变量时,它指向的对象不能改变。但是,引用的对象本身是可以改变的,也就是 对象内部的状态是可以修改的

4. 私有化构造方法。也就是防止外部通过 new 关键字创建多个实例。

5. 最后一个 getSingleton() 方法是提供全局访问点,返回唯一实例。

6. 在类加载时就初始化实例,避免线程安全问题。缺点是无论是否使用该实例,都会创建一个实例,浪费内存资源。

2. 双重检查锁定(懒加载)

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

关键点:

1. 由于 instance 用了 static 修饰(类级别的变量),且没有初始化,那么类加载的时候 instance 赋值为 null。

2. 不加 final ,是因为后续 instance 需要的时候会被赋值;如果加了 final ,那么 instance 永远只能指向 null。当然哈,jdk 是不允许加了 final 的变量为 null 的,会直接编译错误。

3. 加上 volatile 关键字,是保证多线程下的内存可见性。即:一个线程修改了 instance 的值,另一个线程马上就能看到,也就是强制读主内存,不读工作内存。

4. 私有化构造方法。同上,防止外部通过 new 创建多个实例。

getSingleton() 详解:

其实去掉第一个 if 判断也可以,也就是这样:

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

1. 加上第一个 if 的好处是:

无锁判断,instance 不为空直接返回,为 null 再加同步锁,提高性能,避免每次获取都加锁。

2. 加了锁之后为什么还要判断呢?

试想这么一个场景:两个线程同时来了,都发现 instance 为 null,线程A先获取了锁创建了对象,那么线程B获取锁后无需创建对象,所以要在判断一次是否为 null。

3. 静态内部类(懒加载)

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

使用静态内部类的方式实现单例,JVM 会在加载外部类时延迟加载内部类,既能实现懒加载,又能避免多线程问题。

4. 枚举类

public enum Singleton {INSTANCE;public void test() {}
}

其他所有的实现单例的方式其实是有问题的,那就是可能被反序列化和反射破坏。

枚举的写法的优点:

  • 不用考虑懒加载和线程安全的问题,代码写法简洁优雅
  • 线程安全

反编译任何一个枚举类会发现,枚举类里的各个枚举项是是通过static代码块来定义和初始化的,它们会在类被加载时完成初始化,而java类的加载由JVM保证线程安全,所以,创建一个Enum类型的枚举是线程安全的

  • 防止破坏单例

我们知道,序列化可以将一个单例的实例对象写到磁盘,然后再反序列化读回来,从而获得一个新的实例。即使构造函数是私有的,反序列化时依然可以通过特殊的途径去创建类的一个新的实例,相当于调用该类的构造函数。

Java对枚举的序列化作了规定,在序列化时,仅将枚举对象的name属性输出到结果中,在反序列化时,就是通过java.lang.Enum的valueOf来根据名字查找对象,而不是新建一个新的对象。枚举在序列化和反序列化时,并不会调用构造方法,这就防止了反序列化导致的单例破坏的问题。

对于反射破坏单例的而言,枚举类有同样的防御措施,反射在通过newInstance创建对象时,会检查这个类是否是枚举类,如果是,会抛出异常java.lang.IllegalArgumentException: Cannot reflectively create enum objects,表示反射创建对象失败。

5. 模板模式

实现模板方法通常有两步:

1. 抽象类:定义模板方法和抽象方法,在模板方法里会调用抽象方法。

2. 子类:继承抽象类,重写抽象方法。子类运行时调用父类的模板方法,模板方法运行时再去调用子类重写的抽象方法。

源码应用(AQS,Reentrantlock)

1. AQS 的模板方法定义

AQS 是基础的抽象类,提供通用的同步机制。它的 acquire() 和 release() 方法是模板方法。AQS 中的 tryAcquire() 和 tryRelease() 抽象方法,定义了获取锁和释放锁的具体逻辑。

2. ReentrantLock.lock() 源码

这里 ReentrantLock.lock() 内部就是调用 AQS 的模板方法 acquire(),1 表示要获取一个锁,后续 state 会加1。

这是 AQS 的模板方法,其中 tryAcquire(arg) 方法由子类 ReentrantLock 重写实现。

AQS 作为一个抽象类,除了被 ReentrantLock 继承,还被 CountDownLatch、Semaphore继承。所以说,AQS 提供通用的 模板方法,提高了代码的复用性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/896172.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

高清下载油管视频到本地

下载工具并安装: yt-dlp官网地址: GitHub - yt-dlp/yt-dlp: A feature-rich command-line audio/video downloader ffmpeg官网地址: Download FFmpeg 注:记住为其添加环境变量 操作命令: 该指令表示以720p码率下载VIDEO_UR…

Docker挂载数据显式挂载和隐式挂载的区别

项目使用的Docker file 创建数据卷挂载点,结果发现宿主机目录中的数据卷路径下是空的,才知道docker file中创建的数据卷是隐式挂载,并不会在宿主机上留下持久化数据,随着容器被删除隐式挂载的数据卷也会跟着被删除 后面改为在jen…

IOS UITextField 无法隐藏键盘问题

设置UITextField 键盘按钮返回键为“完成”,即return key 设置done .m代码设置代理 //设置代理协议 UITextFieldDelegate, self.mobileTextField.delegate self; ///点击完成键隐藏键盘 - (BOOL)textFieldShouldReturn:(UITextField *)textField{//取…

【深度学习】Unet的基础介绍

U-Net是一种用于图像分割的深度学习模型,特别适合医学影像和其他需要分割细节的任务。如图: Unet论文原文 为什么叫U-Net? U-Net的结构像字母“U”,所以得名。它的结构由两个主要部分组成: 下采样(编码…

RT-Thread+STM32L475VET6实现定时器定时功能

文章目录 前言一、板载资源介绍二、具体步骤1.打开STM32CubeMX进行相关配置1.1 使用外部高速时钟,并修改时钟树1.2 打开定时器(定时器根据自己需求调整)1.3 打开串口1.4 生成工程 2. 配置定时器2.1 打开HWTIMER设备驱动2.2 声明定时器2.3将stm32l4xx_hal_msp.c中HAL…

Linux /etc/fstab文件详解:自动挂载配置指南(中英双语)

Linux /etc/fstab 文件详解:自动挂载配置指南 在 Linux 系统中,/etc/fstab(File System Table)是一个至关重要的配置文件,它用于定义系统开机时自动挂载的文件系统。如果你想让磁盘分区、远程存储(如 NFS&…

链表-基础训练(二)链表 day14

两两交换链表中的节点 题目示意: 给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。 原先我的思路是图像上的思路,但是我感觉还是很复杂…

Unity游戏制作中的C#基础(4)数组声明和使用

一、数组的声明 在 C# 中,声明数组有多种方式,每种方式都有其适用的场景,下面为你逐一详细介绍: 1. 直接初始化声明 这种方式直观且便捷,在声明数组的同时就为其赋初值,让数组从诞生之初就拥有了具体的数据…

【Gin-Web】Bluebell社区项目梳理5:投票功能分析与实现

本文目录 一、投票功能投票流程实现代码redis投票 一、投票功能 投票流程 首先我们要明确,就是 谁(哪个用户:userID) 给 哪个帖子(postID) 投了 什么票(赞成票or反对票)。 赞成票…

XUnity.AutoTranslator-deepseek——调用腾讯的DeepSeek V3 API,实现Unity游戏中日文文本的自动翻译

XUnity.AutoTranslator-deepseek 本项目通过调用腾讯的DeepSeek V3 API,实现Unity游戏中日文文本的自动翻译。 准备工作 1. 获取API密钥 访问腾讯云API控制台申请DeepSeek的API密钥(限时免费)。也可以使用其他平台提供的DeepSeek API。 …

Python爬虫-批量爬取股票数据猫各股票代码

前言 本文是该专栏的第47篇,后面会持续分享python爬虫干货知识,记得关注。 本文笔者以股票数据猫为例子,基于Python爬虫,批量获取各股票代码数据。 具体实现思路和详细逻辑,笔者将在正文结合完整代码进行详细介绍。废话不多说,下面跟着笔者直接往下看正文详细内容。(附…

《Keras 3 :使用 Vision Transformers 进行物体检测》:此文为AI自动翻译

《Keras 3 :使用 Vision Transformers 进行物体检测》 作者:Karan V. Dave 创建日期:2022 年 3 月 27 日最后修改时间:2023 年 11 月 20 日描述:使用 Vision Transformer 进行对象检测的简单 Keras 实现。 (i) 此示例使用 Keras 3 在 Colab 中查看 GitHub 源 介绍 A…

vue-treeselect显示unknown的问题及解决

问题 解决办法 去node-modules包里面找到这个组件的源码,在它dist文件里面找到这个文件,然后搜索unknown,把它删掉就可以解决了。

深入剖析抽象工厂模式:设计模式中的架构利器

深入剖析抽象工厂模式:设计模式中的架构利器 在软件开发领域,设计模式是解决常见问题的通用方案,而抽象工厂模式作为创建型设计模式的重要一员,在构建复杂软件系统时发挥着关键作用。它为创建一系列相关或相互依赖的对象提供了一…

python获取网页内容 靠谱的做法

获取网页内容 response requests.get(url, verifyFalse) 通过这种方式下载网址不太靠谱, 容易出 ssl错误 requests.exceptions.SSLError: HTTPSConnectionPool(hostagri.hainan.gov.cn, port443): Max retries exceeded with url: /hnsnyt/xxgk/gfxwj/index_1.html (Caused by…

MFC中CString的Format、与XML中的XML_SETTEXT格式化注意

1、在MFC中导入 "msxml6.dll",并使用其中的XML_SETTEXT函数,此调用在进行格式化的时候,调用的还是CString.Format()函数! 2、用double类型的数据,格式化整形数%d之前,必须将double强转为int&…

Linux-C-函数栈-SP寄存器

sp(Stack Pointer,栈指针)是计算机体系结构中一个非常重要的寄存器,下面将详细介绍其作用和原理。 作用 1. 管理栈内存 栈是一种后进先出(LIFO,Last In First Out)的数据结构,在程…

从零开始用react + tailwindcs + express + mongodb实现一个聊天程序(一)

项目包含5个模块 1.首页 (聊天主页) 2.注册 3.登录 4.个人资料 5.设置主题 一、配置开发环境 建立项目文件夹 mkdir chat-project cd chat-project mkdir server && mkdir webcd server npm init cd web npm create vitelatest 创建前端项目时我们选择javascrip…

深入理解 QObject的作用

QObject 作为 Qt 库中所有对象的基类,其地位无可替代。几乎 Qt 框架内的每一个类,无论是负责构建用户界面的 QWidget,还是专注于数据处理与呈现的 QAbstractItemModel,均直接或间接继承自 QObject。这种继承体系赋予 Qt 类库高度的…

22爬虫:使用Drission Page的两个案例

案例一:使用DrissionPage抓取BOSS上的招聘信息 使用requests获取BOSS网站上的内容是非常困难的,但是通过网页自动化工具DrissionPage或者是Playwright或者是Seleenium是非常容易的,接下来我们就给出使用DrissionPage爬取BOSS网站python招聘的…