设计模式基础——概述(1/2)

目录

一、设计模式的定义

二、设计模式的三大类别

三、设计模式的原则

四、主要设计模式目录

4.1 创建型模式(Creational Patterns)

4.2 结构型模式(Structural Patterns)

4.3 行为型模式(Behavioral Patterns)


一、设计模式的定义

设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。这些模式代表了最佳实践,通常由有经验的面向对象的软件开发人员采用,以解决在软件设计过程中经常遇到的一般问题。

设计模式提供了一种通用的语言和框架,使得开发者可以更高效地沟通复杂的系统架构,并且允许他们复用已有的解决方案来避免重新发明轮子。每种设计模式都描述了一个在软件开发中不断重复出现的问题,以及解决这个问题的核心思想。它们有助于提高代码的可读性、可维护性和可靠性,并使软件工程更加工程化和规范化。

二、设计模式的三大类别

设计模式分为三大类:创建型模式(关注对象的创建)、结构型模式(关注对象之间的组合)和行为型模式(关注对象之间的交互和职责分配)。

大家在看一些资料的时候,会发现,有人特意提到J2EE类设计模式。这通常指的是在Java 2平台企业版(Java 2 Platform, Enterprise Edition,简称J2EE)中使用的一系列设计模式。这些设计模式是针对开发企业级应用的特殊需求而提出的,它们可能并不属于经典的Gang of Four (GoF)设计模式分类中的三大类:创建型、结构型和行为型。

因此,说“J2EE设计模式”是一种单独的设计模式类别或子集是合理的。这主要是因为J2EE应用程序有其独特的架构特点和挑战,比如处理多层体系结构、分布式计算、事务管理、安全性等。为了解决这些问题,开发者们总结了一系列特定于J2EE环境的最佳实践,这些实践有时被称为“J2EE设计模式”。

需要注意的是,随着技术的发展,特别是Java平台从J2EE到Java EE(Java Platform, Enterprise Edition),再到现在的Jakarta EE(由Eclipse基金会维护的企业级Java规范),一些早期与J2EE相关的模式可能会发生变化或不再适用。同时,新的模式也可能出现以适应新的技术和框架。

比如,大家都熟知的MVC设计模式,这可是初学JAVA必须领会的模式,比如经典的Struts框架,就是基于MVC设计模式。MVC(Model-View-Controller)模式是软件工程中的一种经典架构模式,它在J2EE时代就已经被广泛采用,并且对后续的Web开发框架和企业级应用有着深远的影响。然而,随着技术的发展,尤其是从J2EE到Java EE再到Jakarta EE的过程中,许多新的编程模型和框架出现了,它们提供了不同的方式来组织和管理应用程序的结构。

虽然 MVC 仍然是一种有效的设计原则,但它可能不再像以前那样直接映射到现代 Java 框架中的组件。例如,在Spring框架中,尽管基本的概念类似于MVC,但实现方式却有所差异,比如通过DispatcherServlet处理请求、使用@Controller和@RequestMapping注解来定义控制器行为等。

此外,其他现代前端框架,如AngularJS、React或Vue.js,它们也采用了类似的模式,但具体实现和分工与传统的MVC有所不同。这些框架通常将视图和控制器的职责合并在一起,而模型则负责处理数据和业务逻辑。

因此,说MVC模式不再适合所有情况并不完全准确。它的核心思想——分离关注点,仍然是现代软件设计中的重要原则。只不过,随着时间的推移,这种思想可能会以不同的形式体现出来,或者与其他模式组合起来形成更复杂的架构。开发者需要根据具体的项目需求和技术选型来决定最适合的架构模式。

三、设计模式的原则

以下是几个在软件工程中常用的设计原则:

1. 单一职责原则(Single Responsibility Principle, SRP)
   - 一个类或模块应该只有一个改变的原因。
   - 这个原则强调了将代码的职责划分得尽可能清晰和独立。

2. 开闭原则(Open-Closed Principle, OCP)
   - 对扩展开放,对修改关闭。
   - 这个原则建议通过抽象来实现代码的可扩展性,而尽量避免直接修改现有代码。

3. 里氏替换原则(Liskov Substitution Principle, LSP)
   - 子类型必须能够替换其基类型。
   - 这个原则要求在继承体系中,子类应当可以无限制地替代父类,并且不会导致错误或异常。

4. 接口隔离原则(Interface Segregation Principle, ISP)
   - 客户端不应该被迫依赖于他们不需要的方法。
   - 这个原则提倡创建更小、更专业的接口,而不是大型、通用的接口。

5. 依赖倒置原则(Dependency Inversion Principle, DIP)
   - 高层模块不应该依赖低层模块,两者都应该依赖抽象。
   - 抽象不应该依赖细节,细节应该依赖于抽象。
   - 这个原则有助于提高代码的灵活性和可维护性。

6. 合成复用原则(Composite Reuse Principle, CRP)
   - 尽量使用对象组合,而不是继承来达到复用的目的。
   - 这个原则鼓励使用关联关系来实现代码复用,从而降低耦合度。

7. 迪米特法则(Law of Demeter, LoD)或最少知道原则
   - 一个对象应该对其他对象有尽可能少的了解。
   - 这个原则旨在减少模块之间的耦合度,提高代码的可读性和可维护性。

8. 契约式编程(Programming by Contract, PBC)
   - 类和方法之间应有明确的约定(契约),包括前置条件、后置条件和不变条件。
   - 这个原则可以帮助确保代码的正确性和可靠性。

9. 命令查询分离原则(Command-Query Separation, CQS)
   - 方法要么改变对象的状态(命令),要么返回结果(查询),但不能同时做这两件事。
   - 这个原则有助于提高代码的可预测性和可测试性。

以上这些原则都是为了帮助开发者编写出更加健壮、可读、可维护和可扩展的代码。在实际项目中,需要根据具体场景灵活应用这些原则,并与其他设计原则相平衡。

四、主要设计模式目录

4.1 创建型模式(Creational Patterns)

创建型设计模式是用于处理对象的创建过程,它们将对象的实例化和使用解耦。以下是创建型模式中的一些主要模式:

1. 工厂方法(Factory Method)
   - 定义一个用于创建对象的接口,让子类决定实例化哪一个类。
   - 使得一个类的实例化延迟到其子类。

2. 抽象工厂(Abstract Factory)
   - 提供一个创建一系列相关或相互依赖对象的接口,而无需指定具体的类。
   - 适用于需要提供多个相关的对象系列的情况。

3. 建造者(Builder)
   - 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。
   - 适用于需要生成的产品有多种表现形式或者组装方式时。

4. 原型(Prototype)
   - 提供了一个从现有对象复制新对象的方法,而不是通过new操作符直接创建新的对象。
   - 当系统需要大量创建相似对象,而且创建过程很耗时的时候,适合使用此模式。

5. 单例(Singleton)
   - 确保一个类只有一个实例,并且提供一个全局访问点。
   - 适用于只需要一个实例来协调系统的某些活动的情况。

这些模式的主要目的是隐藏对象的创建细节,使得代码更加灵活和可扩展,同时减少了代码之间的耦合度。在实际开发中,根据具体的需求和场景选择合适的创建型模式可以使代码更加健壮和易于维护。

4.2 结构型模式(Structural Patterns)

结构型设计模式关注于对象和类的组合方式,它们提供了如何将对象组织在一起以便更好地实现系统功能的方法。以下是结构型模式中的一些主要模式:

1. 适配器(Adapter)
   - 将一个类的接口转换成客户希望的另一个接口。
   - 使得原本不兼容的类可以一起工作。

2. 桥接(Bridge)
   - 将抽象部分与它的实现部分分离,使它们都可以独立地变化。
   - 提供了一种将抽象部分和实现部分解耦的方式。

3. 装饰器(Decorator)
   - 动态地给一个对象添加一些额外的职责。
   - 提供一种比继承更有弹性的替代方案来扩展对象的功能。

4. 外观(Facade)
   - 为子系统中的一组接口提供一个一致的界面。
   - 它定义了一个高层接口,使得子系统更容易使用。

5. 享元(Flyweight)
   - 使用共享技术有效地支持大量细粒度的对象。
   - 减少内存使用并提高程序性能。

6. 代理(Proxy)
   - 为其他对象提供一种代理以控制对这个对象的访问。
   - 可以用于创建轻量级的对象或延迟加载、权限检查等场景。

这些模式通过改变对象间的组合关系,提高了代码的复用性、灵活性和可扩展性,并降低了系统的复杂性。在实际开发中,根据具体的需求和场景选择合适的结构型模式可以使代码更加健壮和易于维护。

4.3 行为型模式(Behavioral Patterns)

行为型设计模式关注于对象之间的通信和责任分配,它们描述了如何在类或对象之间划分算法的职责。以下是行为型模式中的一些主要模式:

1. 模板方法(Template Method)
   - 在一个抽象类中定义了一个算法的框架,而将一些步骤延迟到子类中。
   - 它允许子类在不改变结构的情况下重定义算法的特定步骤。

2. 命令(Command)
   - 将一个请求封装为一个对象,从而使用户可以使用不同的请求、队列或者日志请求,同时还可以支持可撤销的操作。
   - 通过将操作与接收者分离,使得新的请求类型易于添加。

3. 策略(Strategy)
   - 定义了一系列算法,并将每一个算法封装起来,让它们可以相互替换。
   - 策略模式让算法的变化独立于使用该算法的客户。

4. 责任链(Chain of Responsibility)
   - 允许将请求沿着处理者链传递,直到被处理为止。
   - 消除请求发送者和接收者之间的耦合关系。

5. 状态(State)
   - 允许对象在内部状态变化时改变其行为。
   - 当控制流基于对象的状态时,使用状态模式更合适。

6. 观察者(Observer)
   - 定义了一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。
   - 提供了一种对象间的一对多发布/订阅机制。

7. 访问者(Visitor)
   - 表示一个作用于某元素结构中的各元素的操作。
   - 它使你可以在不修改各元素类的前提下定义作用于这些元素的新操作。

8. 备忘录(Memento)
   - 在不破坏封装性的前提下,捕获一个对象的内部状态以便稍后恢复它。
   - 提供了一种状态保存和回滚的方法。

9. 中介者(Mediator)
   - 定义一个中介对象来封装一系列的对象交互。
   - 降低了系统的复杂性和耦合度。

10. 迭代器(Iterator)
    - 提供一种顺序访问聚合对象的元素的方式,而不暴露其底层表示。
    - 支持以统一的方式遍历各种数据结构。

11. 解释器(Interpreter)
    - 给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。
    - 当需要实现一个小规模的语言或者表达式解析时,适合使用此模式。

这些模式通过合理地组织对象间的交互,提高了代码的复用性、灵活性和可扩展性,并简化了复杂的系统设计。在实际开发中,根据具体的需求和场景选择合适的行为型模式可以使代码更加健壮和易于维护。

先整理这么多,后面我们有时间详细一起学习几个常用的设计模式。

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

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

相关文章

Vue脚手架 生命周期 组件化开发

Vue脚手架 & 生命周期 & 组件化开发 一、今日目标 1.生命周期 生命周期介绍生命周期的四个阶段生命周期钩子声明周期案例 2.综合案例-小黑记账清单 列表渲染添加/删除饼图渲染 3.工程化开发入门 工程化开发和脚手架项目运行流程组件化组件注册 4.综合案例-小兔…

yolov8 pose coco2yolo

import os import json from tqdm import tqdm import argparseparser argparse.ArgumentParser() # 这里根据自己的json文件位置,换成自己的就行 parser.add_argument(--json_path,defaultrC:\Users\k167\Desktop\dataset\person_dataset/instances_val2017_perso…

MQTT协议对比TCP网络性能测试模拟弱网测试

MQTT正常外网压测数据---时延diff/ms如下图: MQTT弱网外网压测数据 TCP正常外网压测数据 TCP弱网外网压测数据 结论: 在弱网场景下,MQTT和TCP的网络性能表现会有所不同。下面是它们在弱网环境中的对比: 连接建立:M…

【代码随想录刷题】Day20 二叉树06

文章目录 1.【654】最大二叉树1.1 题目描述1.2 解题思路1.3 java代码实现1.4 总结 2.【617】合并二叉树2.1 题目描述2.2 解题思路2.3 java代码实现 3.【700】二叉搜索树中的搜索3.1 题目描述3.2 解题思路3.3 java代码实现 4.【98】验证二叉搜索树4.1 题目描述4.2 解题思路4.3 j…

盘点11月Sui生态发展,了解Sui的近期成长历程!

11月是Web3的“回暖期”,行业持续展现增长趋势。Sui紧随行业脚步,开展了一系列生态活动。其中历时一个多月的Quest 3游戏活动顺利结束并公布奖励,在多地区成功举办Move和Sui生态黑客松&交流会,还有针对中文社区开发者教育的星…

Axure原型图表组件库,数据可视化元件(Axure9大屏组件)

针对Axure制作的大屏图表元件库,帮助产品经理更高效地制作高保真图表原型,是产品经理必备元件工具。现分享完整的组件库,大家一起学习。 本组件库的图表模块,已包含所有常用的图表,以下为部分组件截图示意。文末可下载…

TA-Lib学习研究笔记(九)——Pattern Recognition (2)

TA-Lib学习研究笔记(九)——Pattern Recognition (2) 形态识别的函数的应用,通过使用A股实际的数据,验证形态识别函数,用K线显示出现标志的形态走势,由于入口参数基本上是open, hig…

寒冬不再寒冷:气膜体育馆如何打造温馨运动天地

取暖季即将来临,随着气温逐渐下降,人们在寒冷的冬季里如何保持运动热情和身体的健康成为了一项挑战。而在这个时候,气膜体育馆成为了运动爱好者们的理想场所,提供如春般温暖舒适的运动环境。那么,让我们一起揭秘气膜体…

2024年SEO策略:如何优化您的知识库?

如今很多人在遇到问题时都会求助于谷歌。谷歌已经成为提供解决方案不可或缺的工具。作为全球搜索引擎的巨头,拥有大量用户流量。这就是为什么确保您的产品和服务在谷歌搜索结果中排名靠前是至关重要的,如果您想获得更多的客户,SEO是一个非常关…

Filed II 绘制超声 3D/2D 点扩散函数

点扩散函数可以较好地描述超声对成像目标分辨能力,利用 filed II 仿真工具实现点扩算函数 PSF 的 3D 和 2D 绘制。 定义换能器基本参数 f0=5e6; % Transducer center frequency [Hz] fs=100e6; % Sampling frequency [Hz] c=1540; % Speed of sound [m/s] width=0.15/1000

<Linux> 文件系统

目录 前言: 一、 磁盘 (一)磁盘的物理结构 (二)磁盘的物理存储结构 1. 数据存储 2. 存储结构 二、磁盘的逻辑抽象 三、磁盘信息 (一)具体结构 (二)重新认识目录…

SOLIDWORKS Flow Simulation电子机箱散热

这一次我们来聊聊电子冷却问题,以这个机箱散热问题为例,我们一般的散热设计要求是CPU不能超过80℃,北桥芯片温度不能超过85℃,南桥芯片不超过95℃。在实际情况下芯片内部的各处温度是不一样,面对与芯片级别的散热分析我…

mysql中MDL(元数据锁)的长事务读写阻塞如何解决

MDL,即元数据锁是什么,我们已经介绍过了 那其存在的长事务读写阻塞问题,一般是怎么解决的呢,主要有两种解决方法。 online ddl MySQL5.6开始,推出一项新功能Online DDL,在ALTER或者CREATE INDEX等语句后添…

【教学类-35-05】17号的学号字帖(A4竖版1份)

作品展示: 背景需求: 大四班17号男孩目前无法自主数学数字。他表示自己能够认识数字,但不会写。 保育老师说:我曾经教过他,抓着手示范的。但是他记不住。家里估计也不练习的。年龄还没到,下学期再看看能不…

有限空间作业中毒窒息事故频发,汉威科技创新方案护航

工贸企业有限空间是我国重大事故多发频发的重点领域之一,安全问题形势严峻。 有限空间是指封闭或者部分封闭、未被设计为固定工作场所,人员可以进入,通风不良,易造成有毒有害物质、易燃易爆气体积聚或者氧含量不足的空间&#xf…

excel做预测的方法集合

一. LINEST函数 首先,一元线性回归的方程: y a bx 相应的,多元线性回归方程式: y a b1x1 b2x2 … bnxn 这里: y - 因变量即预测值x - 自变量a - 截距b - 斜率 LINEST的可以返回回归方程的 截距(a) 和 斜…

jsp使用 分页专用工具

分页器,根据过来的参数计算当着页应当从哪一条记录开始显示,并且显示到哪。 PageUtils [pageSize5, currIndex1, totalCount166, totalPage34, startPosition0] PageUtils [pageSize5, currIndex5, totalCount166, totalPage34, startPosition20] PageUt…

5.10 Windows驱动开发:摘除InlineHook内核钩子

在笔者上一篇文章《内核层InlineHook挂钩函数》中介绍了通过替换函数头部代码的方式实现Hook挂钩,对于ARK工具来说实现扫描与摘除InlineHook钩子也是最基本的功能,此类功能的实现一般可在应用层进行,而驱动层只需要保留一个读写字节的函数即可…

得帆云助力容百科技构建CRM系统,实现LTC全流程管理

宁波容百新能源科技股份有限公司 宁波容百新能源科技股份有限公司(以下简称“容百科技”)于2014年9月建立,是高科技新能源材料行业的跨国型集团公司。专业从事锂电池正极材料的研发、生产和销售,于2019年登陆上交所科创板&#x…

Python 数据分析:日期型数据的玩转之道

更多资料获取 📚 个人网站:ipengtao.com 在数据分析的领域中,处理日期型数据是至关重要的一环。Python 提供了丰富的工具和库,使得对日期进行分析、处理、可视化变得更加轻松。本文将深入探讨 Python 中如何玩转日期型数据&#…