常见设计模式之Java实现

引言

设计模式是软件工程中的一套被反复使用的、大家公认的、经过分类编目的代码设计经验的总结。它们是解决特定问题的模板,可以提高代码的可重用性、可读性和可维护性。本文将介绍Java中常见的20种设计模式,并提供具体的使用场景、设计模式的解释以及示例代码。

创建型模式

单例模式(Singleton)

使用场景:需要确保某个类只有一个实例,并且提供一个全局访问点。
解释:单例模式确保一个类只有一个实例,并提供一个全局访问点。
示例代码

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

工厂方法模式(Factory Method)

使用场景:当需要创建对象的类很多时,可以使用工厂方法模式来封装对象的创建过程。
解释:定义一个用于创建对象的接口,让子类决定实例化哪一个类。
示例代码

interface Product {void use();
}class ConcreteProduct implements Product {public void use() {// ...}
}abstract class Creator {abstract Product factoryMethod();
}class ConcreteCreator extends Creator {public Product factoryMethod() {return new ConcreteProduct();}
}

抽象工厂模式(Abstract Factory)

使用场景:当需要创建多个相关或依赖对象的家族时。
解释:提供一个创建一系列相关或相互依赖对象的接口,而不需要指定它们具体的类。
示例代码

// 类似工厂方法模式,但增加了多个产品族的创建

建造者模式(Builder)

使用场景:当创建复杂对象时,需要逐步构建。
解释:将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建不同的表示。
示例代码

class Product {// ...
}class Builder {private Product product = new Product();public Builder setPart1(String part1) {product.setPart1(part1);return this;}// ...public Product build() {return product;}
}

原型模式(Prototype)

使用场景:当创建新对象成本较高时,可以使用原型模式。
解释:通过拷贝现有的实例来创建新的实例,而不是通过新建。
示例代码

class Prototype implements Cloneable {// ...public Object clone() {try {return super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return null;}
}

结构型模式

适配器模式(Adapter)

使用场景:当需要将一个类的接口转换成客户期望的另一个接口时。
解释:使原本由于接口不兼容而不能一起工作的类可以一起工作。
示例代码

interface Target {void request();
}class Adaptee {public void specificRequest() {// ...}
}class Adapter implements Target {private Adaptee adaptee;public Adapter(Adaptee adaptee) {this.adaptee = adaptee;}public void request() {adaptee.specificRequest();}
}

装饰器模式(Decorator)

使用场景:当需要动态地给一个对象添加额外的职责时。
解释:在不修改对象结构的情况下,动态地给对象添加职责。
示例代码

interface Component {void operate();
}class ConcreteComponent implements Component {public void operate() {// ...}
}abstract class Decorator implements Component {protected Component component;public Decorator(Component component) {this.component = component;}public void operate() {component.operate();}
}class ConcreteDecoratorA extends Decorator {public ConcreteDecoratorA(Component component) {super(component);}public void operate() {super.operate();// 添加额外职责}
}

代理模式(Proxy)

使用场景:当需要控制对原始对象的访问时。
解释:为其他对象提供一个代替或占位符作为它的接口。
示例代码

interface Subject {void request();
}class RealSubject implements Subject {public void request() {// ...}
}class Proxy implements Subject {private RealSubject realSubject;public Proxy() {this.realSubject = null;}public void request() {if (realSubject == null) {realSubject = new RealSubject();}realSubject.request();}
}

外观模式(Facade)

使用场景:当需要提供一个客户端可以访问系统的复杂接口的简化接口时。
解释:定义一个高层接口,使得子系统更容易使用。
示例代码

interface SubSystem {void operation();
}class SubSystem0 implements SubSystem {public void operation() {// ...}
}class SubSystem1 implements SubSystem {public void operation() {// ...}
}class Facade {private SubSystem0 subSystem0 = new SubSystem0();private SubSystem1 subSystem1 = new SubSystem1();public void operation() {subSystem0.operation();subSystem1.operation();}
}

桥接模式(Bridge)

使用场景:当需要将抽象部分与它的实现部分分离,使它们可以独立地变化时。
解释:将抽象与实现解耦,让它们可以独立地变化。
示例代码

abstract class Abstraction {protected Implementor implementor;public Abstraction(Implementor implementor) {this.implementor = implementor;}public void operation() {implementor.operationImpl();}
}interface Implementor {void operationImpl();
}class ConcreteImplementorA implements Implementor {public void operationImpl() {// ...}
}class RefinedAbstraction extends Abstraction {public RefinedAbstraction(Implementor implementor) {super(implementor);}// ...
}

组合模式(Composite)

使用场景:当需要将对象组合成树形结构以表示“部分-整体”的层次结构时。
解释:将对象组合成树形结构来表示“部分-整体”的层次结构。
示例代码

interface Component {void operation();
}class Leaf implements Component {public void operation() {// ...}
}class Composite implements Component {private List<Component> children = new ArrayList<>();public void add(Component component) {children.add(component);}public void operation() {for (Component component : children) {component.operation();}}
}

享元模式(Flyweight)

使用场景:当需要减少创建大量相似对象时。
解释:通过共享来高效地支持大量细粒度的对象。
示例代码

class Flyweight {private String intrinsicState;public Flyweight(String intrinsicState) {this.intrinsicState = intrinsicState;}public void operation(String extrinsicState) {// ...}
}class FlyweightFactory {private HashMap<String, Flyweight> flyweights = new HashMap<>();public Flyweight getFlyweight(String key) {if (!flyweights.containsKey(key)) {flyweights.put(key, new Flyweight(key));}return flyweights.get(key);}
}

行为型模式

策略模式(Strategy)

使用场景:当需要在运行时选择算法或行为时。
解释:定义一系列算法,把它们一个个封装起来,并使它们可互换。
示例代码

interface Strategy {void algorithmInterface();
}class ConcreteStrategyA implements Strategy {public void algorithmInterface() {// ...}
}class Context {private Strategy strategy;public Context(Strategy strategy) {this.strategy = strategy;}public void executeStrategy() {strategy.algorithmInterface();}
}

模板方法模式(Template Method)

使用场景:当需要在方法中定义算法的框架,将一些步骤的实现延迟到子类中时。
解释:定义算法的骨架,将一些步骤的实现延迟到子类中。
示例代码

abstract class AbstractClass {public void templateMethod() {step1();step2();step3();}public abstract void step2();public void step1() {// ...}public void step3() {// ...}
}class ConcreteClass extends AbstractClass {public void step2(){// ...}
}

观察者模式(Observer)

使用场景:当一个对象的改变需要同时改变其他对象,而不知道具体有多少对象待改变时。
解释:对象间存在一对多关系时,则使用观察者模式。
示例代码

interface Observer {void update();
}interface Subject {void registerObserver(Observer o);void removeObserver(Observer o);void notifyObservers();
}class ConcreteSubject implements Subject {private List<Observer> observers = new ArrayList<>();public void registerObserver(Observer o) {observers.add(o);}public void removeObserver(Observer o) {observers.remove(o);}public void notifyObservers() {for (Observer observer : observers) {observer.update();}}
}class ConcreteObserver implements Observer {public void update() {// ...}
}

迭代器模式(Iterator)

使用场景:当需要访问一个聚合对象中的元素,而又不暴露其内部的表示时。
解释:提供一种顺序访问一个聚合对象中的各个元素的方法,而又不暴露其内部的表示。
示例代码

interface Iterator {boolean hasNext();Object next();
}interface Aggregate {Iterator createIterator();
}class ConcreteIterator implements Iterator {private List items;private int position = 0;public ConcreteIterator(List items) {this.items = items;}public boolean hasNext() {return position < items.size();}public Object next() {return items.get(position++);}
}class ConcreteAggregate implements Aggregate {private List items = new ArrayList();public Iterator createIterator() {return new ConcreteIterator(items);}
}

中介者模式(Mediator)

使用场景:当系统中对象之间存在复杂的引用关系时。
解释:用一个中介对象来封装一系列对象之间的交互。
示例代码

interface Mediator {void register(String colleague, Colleague colleague);void relay(String colleague);
}interface Colleague {void setMediator(Mediator mediator);void notify(String message);
}class ConcreteMediator implements Mediator {private Map<String, Colleague> colleagues = new HashMap<>();public void register(String colleague, Colleague colleague) {colleagues.put(colleague, colleague);}public void relay(String colleague, String event) {Colleague c = colleagues.get(colleague);// ...}
}class ConcreteColleagueA implements Colleague {private Mediator mediator;public void setMediator(Mediator mediator) {this.mediator = mediator;}public void notify(String message) {mediator.relay("ColleagueA", message);}
}

命令模式(Command)

使用场景:当需要将请求封装为一个对象,从而使用不同的请求、队列或日志请求时。
解释:将请求封装为一个对象,从而使你可用不同的请求、队列或日志来参数化其他对象。
示例代码

interface Command {void execute();
}class Receiver {public void action() {// ...}
}class ConcreteCommand implements Command {private Receiver receiver;public ConcreteCommand(Receiver receiver) {this.receiver = receiver;}public void execute() {receiver.action();}
}class Invoker {private Command command;public void setCommand(Command command) {this.command = command;}public void executeCommand() {command.execute();}
}

责任链模式(Chain of Responsibility)

使用场景:当多个对象处理请求,但具体哪个对象处理该请求待运行时才能确定时。
解释:使多个对象都有机会处理请求。从而避免请求的发送者和接收者之间的耦合关系。
示例代码

interface Handler {void handleRequest(Request request);
}class ConcreteHandler1 extends Handler {private Handler next;public void setNext(Handler next) {this.next = next;}public void handleRequest(Request request) {if (next != null) {next.handleRequest(request);}}
}class Request {// ...
}

备忘录模式(Memento)

使用场景:当需要保存和恢复对象的内部状态时。
解释:在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。
示例代码

class Memento {private String state;public Memento(String state) {this.state = state;}public String getState() {return state;}
}class Originator {private String state;public Memento saveStateToMemento() {return new Memento(state);}public void getStateFromMemento(Memento memento) {state = memento.getState();}
}class Caretaker {private Memento memento;public void setMemento(Memento memento) {this.memento = memento;}public Memento getMemento() {return memento;}
}

状态模式(State)

使用场景:当一个对象的行为取决于它的状态,并且它的状态值在运行时改变时。
解释:允许对象在内部状态改变时改变它的行为。
示例代码

interface State {void handle(StateContext context);
}class ConcreteStateA implements State {public void handle(StateContext context) {// ...}
}class StateContext {private State state;public StateContext(State state) {this.state = state;}public void handleState() {state.handle(this);}
}

访问者模式(Visitor)

使用场景:当需要对一个对象结构中的对象进行操作,而又不想让这些操作依赖于对象的类时。
解释:为一个对象结构(比如组合结构)增加新能力。
示例代码

interface Element {void accept(Visitor visitor);
}class ConcreteElementA implements Element {public void accept(Visitor visitor) {visitor.visit(this);}
}interface Visitor {void visit(ConcreteElementA element);
}class ConcreteVisitor implements Visitor {public void visit(ConcreteElementA element) {// ...}
}

结语

本文介绍了Java中常见的20种设计模式,每种模式都提供了使用场景、简要的解释以及示例代码。设计模式是软件开发中的重要工具,它们帮助我们写出更加清晰、灵活且可维护的代码。希望本文能够帮助你更好地理解和应用设计模式。

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

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

相关文章

高通 Android 12/13冻结屏幕

冻结屏幕很多第一次听到以为是Android一种异常现象&#xff0c;实则不然&#xff0c;就是防止用户在做一些非法操作导致问题防止安全漏洞问题。 1、主要通过用户行为比如禁止下拉状态栏和按键以及onTouch事件拦截等&#xff0c;不知道请看这篇文章&#xff08;Touch事件传递流…

GitHub打不开的解决方案

1、打开https://sites.ipaddress.com/github.com/找到DNS Resource Records&#xff0c;复制github的ip地址&#xff0c;先保存起来&#xff1a; 140.82.112.32、打开https://sites.ipaddress.com/fastly.net/找到DNS Resource Records&#xff0c;复制其中一个ip地址&#xf…

基于Nacos实现Sentinel规则持久化

基于Nacos实现Sentinel规则持久化 一、Sentinel使用痛点二、解决方案2.1 保存本地文件2.2 保存数据库2.3 保存到Nacos 三、规则持久化到Nacos3.1 Nacos服务端修改配置3.2 Sentinel控制台修改配置3.3 Nacos数据源整合到Sentinel中 一、Sentinel使用痛点 SpringCloudAlibaba帮我…

迷你手持小风扇哪个牌子质量好点?这五款迷你手持小风扇不要错过

随着空调的普及&#xff0c;我们对夏日热浪的抵抗力逐渐减弱。当从凉爽的空调屋步入闷热的户外、拥挤的交通工具或公共场所时&#xff0c;如何抵御热浪的侵袭成为大众关注的焦点。在这样的背景下&#xff0c;迷你手持小风扇凭借其便携性和即时降温功能&#xff0c;成为众多人的…

341_C++_使用C++中的std::map容器查找键的时候,注意给个默认值,防止成员函数value()未查找到键,确保即使键不存在,你也能安全地获取一个值

在C++中,std::map是一个关联容器,它存储的元素都是键值对(key-value pairs),并且按键(key)自动排序。 std::map有一个成员函数value(),它用于返回与给定键相关联的值。如果该键在map中不存在,则value()函数返回提供的默认值。 map<int, int> mKeyMaps; int ke…

降价!免费!AI大模型开启价格战,企业如何“薅”出绿色财富?

近期&#xff0c;国内大模型技术供应商之间的价格战&#xff0c;使得这项原本成本较高的技术变得更加亲民&#xff0c;极大降低了企业的技术采用门槛。这不仅为企业提供了经济实惠的技术解决方案&#xff0c;更为他们的绿色低碳转型之路带来了新的机遇。 随着全球气候变化问题…

MySQL -- 相关知识点

1.数据库相关介绍 数据库的选择通常取决于具体的应用需求&#xff0c;如性能、扩展性、数据一致性和易用性等因素。 1. 关系型数据库&#xff08;RDBMS&#xff09; MySQL&#xff1a; 广泛使用的开源数据库&#xff0c;支持大多数操作系统。强调易用性、灵活性和广泛的社区支…

基于Java+MySQL+Swing的学生管理系统

1.系统简介和开发背景 该同学工作积极主动、高效&#xff0c;学习认真&#xff0c;待人诚恳&#xff0c;能够做到服从指挥、认真听取老同志的指导&#xff0c;不怕苦、不怕累&#xff0c;表现有较强的求知欲&#xff0c;积极观察、体验、思考&#xff0c;并能够灵活运用自己的知…

自养号测评是什么?亚马逊、沃尔玛、Target卖家如何建立自己的护城河?

近期有跨境卖家咨询我自养买家账号测评的事情&#xff0c;他们还是有不了解自养号测评的&#xff0c;所以珑哥觉得有必要再讲一下卖家测评的一些事情&#xff0c;之前文章也说过。这可能是跨境卖家运营的一个趋势。今天珑哥着重来介绍一下自养号测评 一、什么叫做自养号测评&a…

OrangePi AIpro初体验之图片视频检测案例真实测评

OrangePi AIpro简介 OrangePi AIpro官网 Orange Pi AI Pro 开发板是香橙派联合华为精心打造的高性能AI 开发板&#xff0c;其搭载了昇腾AI 处理器&#xff0c;可提供8TOPS INT8 的计算能力&#xff0c;内存提供了8GB 和16GB两种版本。可以实现图像、视频等多种数据分析与推理…

简述v-for中的key的理解

在 Vue 中&#xff0c;v-for 是用来基于一个数组来渲染一个列表的指令。而 key 是一个特殊的属性&#xff0c;在 v-for 中经常被用到&#xff0c;它主要帮助 Vue 跟踪每个节点的身份&#xff0c;从而重用和重新排序现有元素。 key 的作用 性能优化&#xff1a;当 Vue 检测到列…

2024-python字典-报错compounds.iloc[0].molecule_structures.keys()

2024-python字典-报错compounds.iloc[0].molecule_structures.keys() .keys()拿不到 需要如何解决呢 import ast# 假设 "molecular_structures "是一个看起来像 Python 字典的字符串 molecule_structures df_VEGFR2_compounds.iloc[0].molecule_structures # 安全…

全球点赞最高的人颜廷利:真正的人生目标是什么

在那个充满生机的2024年春天&#xff0c;记者有幸对中国第一起名大师的老师颜廷利教授进行了深入的访谈。带着对其人生哲学的强烈好奇&#xff0c;记者紧张而期待地提出了问题&#xff1a;“颜教授&#xff0c;您在漫长的人生旅途中最追求的是什么&#xff1f;” 宁夏银川、山东…

PMP考试没有考过应该如何再考?

一、接受失败&#xff0c;理性分析 其实&#xff0c;PMP考试可以在PMI一年有效期内提交补考申请&#xff0c;若已经过了一年有效期&#xff0c;考生则需要重新进行完整的PMP考试报名流程。PMP考试补考费用&#xff1a;2500元/次。 1.接受失败&#xff1a;接受并处理情绪。面对…

【Python数据分析】基于自回归积分滑动平均模型的疫情分析报告 附完整python代码

资源地址&#xff1a;Python数据分析大作业 2000字 图文分析文档 疫情分析完整python代码 数据分析 数据来自法国疫情数据 资源地址&#xff1a;Python数据分析大作业 2000字 图文分析文档 疫情分析完整python代码 代码详解 完整代码文件 主要是对时间序列数据进行分析和预…

C++|四种方法解决三线程按顺序轮流打印ABC的问题:互斥锁和条件变量、原子操作、信号量

基于Pthreads线程库的实现 g your_program.cpp -lpthread编译时确保链接了Pthreads库. #include <pthread.h> #include <iostream> #include <unistd.h>// 全局变量&#xff0c;用于控制打印顺序 int turn 0; // 定义互斥锁和条件变量 pthread_mutex_t lo…

【#4 如何选择激活函数:深度学习中的关键决策】

文章目录 前言第一部分&#xff1a;激活函数的基础知识1.1 激活函数简介1.2 激活函数的作用 第二部分&#xff1a;选择激活函数的考虑因素2.1 网络层级2.2 任务类型2.3 训练稳定性2.4 计算效率2.5 网络深度 第三部分&#xff1a;流行激活函数的特点与应用场景3.1 Sigmoid3.2 Re…

技术面‍:前端代码是如何与服务器交互的

前言&#xff1a; 本篇文章主要是想讲解 .html 文件和 .CSS 文件在实际开发中和后端服务器交互最后上线的基础原理。 面向的人群&#x1f195;&#xff1a;是刚入行不久&#xff0c;且目前只会写前端业务代码而不清楚整个工作流的前端新人。我会从 0 开始一步一步带你理解整个…

QT打包命令的使用

QT打包命令的使用 windeployqt untitled1.exe windeployqt -qmldir C:\Users\unikinfo\Documents\UNIKINFO\untitled\YNMapDisplay YNMapDisplay.dll linuxdeployqt 程序文件 -qmldirqml源码路径 -appimage

大规模敏捷SA(Leading SAFe)证书是什么意思?如何报名,含金量高吗?

大规模敏捷SA(Leading SAFe)证书是什么意思&#xff1f; 常规的敏捷框架适用于中小型项目团队&#xff0c;而且不具有扩展性。基于常规的敏捷框架&#xff0c;SAFe定义了一个可扩展的敏捷框架模型&#xff0c;它适用于大型团队的合作开发&#xff0c;可以提高团队之间的协作性…