【零】Java设计模式GOF23之总结

2024-01-29 13:50
文章标签 java 总结 设计模式 gof23

本文主要是介绍【零】Java设计模式GOF23之总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

设计模式主要有3个大分类

创建型模式:用来建造对象

一、单例模式:

保证一个类只有一个实例,并且提供一个访问该实例的全局访问点。

常见应用场景

1.全局计数器采用单例模式,不然不好同步。

2.应用程序的日志应用,共享日志文件一直处于打开状态,所以只能有一个实例去操作,否则内容不好追加。

3.数据库连接池的设计也用单例,否则浪费资源。

4.spring中的bean默认都是单例。

5.servlet是单例。

6.spring mvc/ structs1,控制器对象是单例。

7.项目中工具类,一般都用单例,没必要浪费资源。

常见实现方式

1.饿汉模式(线程安全、调用效率高、不能延时加载)

2.懒汉模式(线程安全、调用效率不高(getInstance方法用了synchronized同步方法)、可以延时加载)

3.静态内部类模式(线程安全、调用率高、可以延时加载)

4.双重同步锁单例模式

5.枚举单例(线程安全、调用率高、不能延时加载、不会被反射反序列化生成多个实例

代码

各种单例模式代码演示

UML图

二、工厂模式

选择实现类、创建对象统一管理和控制。

实现了创建者和调用者的分离

用工厂方法代替new操作。

常见应用场景

1.JDK中Calendar的getInstance方法

2.JDBC中Connection对象的获取

3.Hibernate中SessionFactory创建session对象

4.spring中IOC容器创建管理bean对象

5.反射中Class对象的newInstance

6.JDK中java.util.concurrent包中的Executors类,就是一个静态工厂,用来创建各种线程池。代码示例用这个案例

1.简单工厂模式

用来生产同一等级结构中的任意产品。(对于增加新的产品,需要修改已有代码

代码

【十】Java设计模式GOF23之简单工厂模式(静态工厂模式)

UML图

源码中间多了一层抽象AbstractExecutorService,在普通的静态工厂中是不需要这层抽象的。

2.工厂方法模式

用来生产同一等级结构中的固定产品。(支持增加任意产品)

代码

【十一】Java设计模式GOF23之工厂方法模式

UML图

3.抽象工厂模式

用来生产不同产品族的全部产品。(对于增加新的单个产品,无能为力,支持增加产品族)

它是工厂方法模式的升级版,在有多个业务品种、业务分类时,通过抽象工厂模式产生需要的对象是一种非常好的解决方式。

代码

【十二】Java设计模式GOF23之抽象工厂模式

UML图

三、建造者模式

分离了对象子组件的单独构造(有builder来负责)和装配(由Director负责)。从而可以构造出复杂的对象。

这个模式适用于:某个对象的构建过程复杂的情况下。

由于实现了构建和装配的解耦。不同的构建器,相同的装配,也可以做出不同的对象;相同的构建器,不同的装配顺序也可以做出不同的对象。

也就是实现了构建算法、装配算法的解耦,实现了更好的复用。

关键代码:建造者:创建和提供实例,导演:管理建造出来的实例的依赖关系。

常见应用场景

1.SpringBuilder类的append方法

2.SQL中的PreparedStatement

3.JDOM中的DomBuilder、SAXBuilder

4.去肯德基,汉堡、可乐、薯条、炸鸡翅等是不变的,而其组合是经常变化的,生成出所谓的"套餐"。

之前遇到一个笔试题应该就是考构建者模式的,例子是说的星巴克。“咖啡价格问题:星巴克咖啡有基本的美式,摩卡,拿铁三种,价格都不一样。同时有很多的配料,包括香草、肉桂、奶油、焦糖、巧克力,每种配料的价格也不一样。一种咖啡可以选多种配料。以OOD的方式来计算每杯咖啡的价格。”

代码

【五】Java设计模式GOF23之建造者模式

UML图

四、原型模式prototype

如果通过new创建一个对象需要非常繁琐的数据准备和访问权限,则可以用原型模式。

就是java中的克隆,以某个对象为原型,复制出新对象。新对象具备原型对象的特点。

优势:效率高(直接克隆,避免了重新执行构造过程步骤)

代码

【十四】Java设计模式GOF23之原型模式

结构型模式:

一、适配器模式

将一个类的接口转换成客户希望的另外一个接口。

适配器不是在详细设计时添加的,而是解决正在服役的项目的问题。一个系统如果太多出现这种情况,无异于一场灾难。

类适配器用继承实现

对象适配器用依赖实现

代码

【十】Java设计模式GOF23之适配器模式(类方式实现、对象方式实现)

个人思考:这个设计模式不是很鸡肋吗,还是我没理解到它的精髓?

直接用adaptee去实现target接口,在request方法中写自己的specificRequest方法的逻辑就好了啊,为毛要多个adapter类出来?

什么毛病?

UML图

二、桥接模式

它的主要特点是把抽象(abstraction)与行为实现(implementation)分离开来,从而可以保持各部分的独立性以及应对它们的功能扩展

适用处理多维度变化。

举个例子,引用例子Java 设计模式之桥接模式(七)

 

这个例子里面,电脑类型是一个维度,电脑品牌是一个维度,把品牌放在类型下面,那么类型的增加会导致品牌的增加。

品牌的增加,也会去每个类型下面增加一个对象。

桥接模式的意思就是,以上面这个例子来说,应该提两个维度的接口或父类出来,他们之间可以考虑依赖的方式。这样一个维度变更不影响另外一个维度。比如:

三、装饰模式

四、组合模式

五、外观模式

六、享元模式

七、代理模式

核心作用:通过代理,控制对对象的访问

可以详细控制访问某个(某类)对象的方法,在调用这个方法前做前置处理,调用这个方法后做后置处理。(Spring AOP的围观实现)。从而实现将统一流程代码放到代理类中处理。

应用场景

1.安全代理:屏蔽对真实角色的直接访问

2.远程代理:通过代理类处理远程方法调用RMI

3.延迟加载:先加载轻量级的代理对象,真正需要时再加载真实对象

1.静态代理

自己定义代理类

代码

【十五】Java设计模式GOF23之静态代理

UML图

2.动态代理

动态生成代理类

代码

这里使用反射来实现的

【十六】Java设计模式GOF23之动态代理

UML图

行为型模式:

一、模板方法模式

java.util.AbstractQueuedSynchronizer类的设计就使用了模板方法模式。

代码

父类AQS,写了释放锁的逻辑release()方法,

而release()方法中有部分逻辑是需要调用tryRelease()方法实现:

    public final boolean release(int arg) {if (tryRelease(arg)) {Node h = head;if (h != null && h.waitStatus != 0)unparkSuccessor(h);return true;}return false;}

父类AQS并没有实现tryRelease()方法,而是由子类Sync去实现自己的tryRelease()方法。

父类描述了整体的释放锁的逻辑,部分细节由子类描述,不同的子类有不同的描述。

比如ReentrantLock中的Sync的tryRelease()

protected final boolean tryRelease(int releases) {int c = getState() - releases;if (Thread.currentThread() != getExclusiveOwnerThread())throw new IllegalMonitorStateException();boolean free = false;if (c == 0) {free = true;setExclusiveOwnerThread(null);}setState(c);return free;}

countDownLatch中的Sync的tryRelease()

        protected boolean tryReleaseShared(int releases) {// Decrement count; signal when transition to zerofor (;;) {int c = getState();if (c == 0)return false;int nextc = c-1;if (compareAndSetState(c, nextc))return nextc == 0;}}

【二十一】Java多线程J.U.C之AQS框架源码导读(总结、干货)(模板模式template pattern)

UML图

二、命令模式

命令模式(Command Pattern)是一种数据驱动的设计模式,它属于行为型模式。请求以命令的形式包裹在对象中,并传给调用对象。调用对象寻找可以处理该命令的合适的对象,并把该命令传给相应的对象,该对象执行命令。

JDK的java.util.concurrent包的ThreadPoolExecutor类就用到命令模式。

简易命令模式组成
1.Command(抽象命令类): 声明了用于执行请求的的exceute()等方法

2.ConcreteCommand(具体命令类): 抽象命令类的子类,对应具体的接收者对象,将接收者对象的动作绑定其中。在实现execute()方法时,将调用接收者对象的相关操作(Action)。

3.Invoker(调用者): 调用命令对象执行请求,它持有了一个命令队列,客户端代码可以向它提交要执行的任务(命令)。

4.Receiver(接收者): 真正的命令执行对象。任何一个类都可以成为接收者,实施和执行请求的方法叫做行动方法。
UML图

代码 

【十三】Java设计模式GOF23之命令模式(例子ThreadPoolExecutor类)

三、迭代器模式

简介

迭代器模式(Iterator Pattern)是 Java 和 .Net 编程环境中非常常用的设计模式。这种模式用于顺序访问集合对象的元素,不需要知道集合对象的底层表示。

它提供一种方法顺序访问一个聚合对象中各个元素, 而又无须暴露该对象的内部表示。

主要成员

1.抽象容器:一般是一个接口,提供一个iterator()方法,例如java中的Collection接口,List接口,Set接口等。
2.具体容器:就是抽象容器的具体实现类,比如List接口的有序列表实现ArrayList,List接口的链表实现LinkList,Set接口的哈希列表的实现HashSet等。
3.抽象迭代器:定义遍历元素所需要的方法,一般来说会有这么三个方法:取得第一个元素的方法first(),取得下一个元素的方法next(),判断是否遍历结束的方法isDone()(或者叫hasNext()),移出当前对象的方法remove(),
4.迭代器实现:实现迭代器接口中定义的方法,完成集合的迭代。
UML图

代码

【十四】Java设计模式GOF23之迭代器模式

四、观察者模式

主要用于1:N的通知

应用场景:聊天室。发布/订阅。网络游戏。消息广播。事件监听。

多个订阅者称为观察者。需要同步给多个订阅者的数据封装到对象中,称为目标。

代码

【十七】Java设计模式GOF23之观察者模式

UML图

五、中介者模式

六、备忘录模式

七、解释器模式

八、状态模式

九、策略模式

在策略模式(Strategy Pattern)中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为型模式。

在策略模式中,我们创建表示各种策略的对象和一个行为随着策略对象改变而改变的 context 对象。策略对象改变 context 对象的执行算法。

策略模式的角色和职责

1.Strategy:策略(算法)抽象

2.ConcreteStrategy:各种策略(算法)的具体实现

3.Context:策略的外部封装类,或者说策略的容器。根据不同的策略执行不同的行为。策略由外部环境决定。

java.util.concurrent中的ThreadPoolExecutor类在实现拒绝策略的时候就是用的策略模式
代码

【十二】Java设计模式GOF23之策略模式(ThreadPoolExecutor类的拒绝策略源码)

UML图

十、责任链模式

责任链模式(Chain of Responsibility Pattern)为请求创建了一个接收者对象的链。这种模式给予请求的类型,对请求的发送者和接收者进行解耦。这种类型的设计模式属于行为型模式。

在这种模式中,通常每个接收者都包含对另一个接收者的引用。如果一个对象不能处理该请求,那么它会把相同的请求传给下一个接收者,依此类推。

javax.servlet.Filter就是用的责任链模式
代码

【十一】Java设计模式GOF23之责任链模式

UML图

十一、访问者模式

这篇关于【零】Java设计模式GOF23之总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/657125

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Python版本与package版本兼容性检查方法总结

《Python版本与package版本兼容性检查方法总结》:本文主要介绍Python版本与package版本兼容性检查方法的相关资料,文中提供四种检查方法,分别是pip查询、conda管理、PyP... 目录引言为什么会出现兼容性问题方法一:用 pip 官方命令查询可用版本方法二:conda 管理包环境方法

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input