软件设计模式(二):工厂、门面、调停者和装饰器模式

news2024/9/20 20:21:58

前言

        在这篇文章中,荔枝将会梳理软件设计模式中的四种:工厂模式、Facade模式、Mediator模式和装饰器Decorator模式。其中比较重要的就是工厂模式和装饰器模式,工厂模式在开发中使用的频数比较高。希望荔枝的这篇文章能讲清楚哈哈哈哈,希望能帮助到有需要的小伙伴~~~


文章目录

前言

一、工厂模式

1.1 简单工厂

1.2 工厂方法

1.3 抽象工厂

二、调停者模式和门面模式

2.1 门面模式Facade

2.2 调停者模式Mediator

三、装饰器模式Decorator 

总结


一、工厂模式

1.1 简单工厂

首先定义一个交通工具的工厂函数接口Moveable。

package com.crj.factorymethod;
/**
 * 工厂函数接口
 */
public interface Moveable {
    void go();
}

 我们要实现某一种方式的交通工具就需要通过将该交通工具类实现该接口并重写go方法。

package com.crj.factorymethod;

/**
 * Car类继承工厂函数接口
 */
public class Car implements  Moveable {
    public void go() {
        System.out.println("Car go wuwuwuwuw....");
    }
}

通过简单工厂模式来返回实例化后的交通工具对象。

package com.crj.factorymethod;

/**
 * 简单工厂模式
 * 简单工厂的可扩展性不好
 */
public class SimpleVehicleFactory {
    public Car createCar() {
        //before processing
        return new Car();
    }

    public Broom createBroom() {
        return new Broom();
    }
}

1.2 工厂方法

        上文中可以看出,简单工厂模式的可扩展性比较差,但我们需要新创建一种交通方式的时候就需要重新在工厂SimpleVehicleFactory定义并写死相应的对象方法,这在后期的时候并不容易维护。那么还可以通过不同交通工具各自有一个工厂,方法类型都是继承自Moveable,返回值也是Moveable类型。

package com.crj.factorymethod;

/**
 * 工厂方法
 */
public class CarFactory {
    public Moveable create() {
        System.out.println("a car created!");
        return new Car();
    }
}

main方法

package com.crj.factorymethod;

public class Main {
    public static void main(String[] args) {
        Moveable m = new CarFactory().create();
        m.go();
    }
}

1.3 抽象工厂

前面我们已经把产品的构建和工厂一一对应了,但是我们操作起产品簇还不是很方便,我们可以通过抽象工厂直接操作创建一整个产品簇。在实现创建产品簇之前,我们需要创建抽象产品类、抽象工厂、具体产品类和具体的工厂。

抽象产品类

首先定义一些抽象产品类,并定义相应的抽象方法。

抽象工厂AbstractFactory

抽象工厂定义抽象的方法用于创建相应的产品簇中的类型产品。 

package com.mashibing.dp.abstractfactory;

public abstract class AbastractFactory {
    abstract Food createFood();
    abstract Vehicle createVehicle();
    abstract Weapon createWeapon();
}

具体工厂类

具体工厂类继承自抽象工厂类,并重写不同类型的产品创建产品的方法。 

具体产品类

具体的产品类继承自抽象产品类

main文件

        在main文件中我们通过实例化具体产品类得到对象并调用相应的产品创建方法首先产品簇的创建。可以看到在这段demo中我们即使需要得到不同的产品簇,也仅需要通过实例化不同的工厂类即可。

package com.mashibing.dp.abstractfactory;

public class Main {
    public static void main(String[] args) {
        AbastractFactory f = new ModernFactory();

        Vehicle c = f.createVehicle();
        c.go();
        Weapon w = f.createWeapon();
        w.shoot();
        Food b = f.createFood();
        b.printName();
    }
}

具体的逻辑可以看一下这张图,荔枝感觉还是没讲得很清楚,剩下三分自己就得靠大家聪明得脑瓜啦。


二、调停者模式和门面模式

2.1 门面模式Facade

        门面模式要求一个系统外部与其内部的通信必须通过一个统一的对象进行,通过一个高层次的接口管理外部通信与内部对象簇之间的通信,简单理解就是一个中介(门面),外部仅需要跟这个中介对接即可,中介再委派客户端发送的请求到相应的子系统中,这也是一种黑箱操作。在该模式下可以同时拥有一个或者多个系统,每个系统是一个类的集合,同时系统对于门面只是当成客户端来处理。

需要注意的是:

  • 一个系统可以有多个门面

  • 门面不参与系统内的业务逻辑

2.2 调停者模式Mediator

        调停者模式相当于一个星型连接,该模式多用于消息中间件如MQ中,对象间只需要跟中间人进行沟通即可,由中间人进行消息的收发和处理,是一种解耦的操作。对于门面模式和调停者模式的区别详情可以下图,二者其实区别并不大并且在某些场景下是可以互换的。


三、装饰器模式Decorator 

装饰器(Decorator)模式的定义:指在不改变现有对象结构的情况下,动态地给该对象增加一些功能的模式,它属于对象结构型模式。

优点:

  • 装饰器是继承的有力补充,比继承灵活,在不改变原有对象的情况下,动态的给一个对象扩展功能,即插即用
  • 通过使用不用装饰类及这些装饰类的排列组合,可以实现不同效果
  • 装饰器模式完全遵守开闭原则

缺点:装饰器模式会增加许多子类,过度使用会增加程序复杂性。

大致看了一些概念性的东西大机器是不是感觉不是特别具象,下面我们可以通过一个例子来理解一下装饰器模式具体是一个怎样的软件设计模式。 

package com.crj.Decorator;

public class Decorator {
    /**
     * 装饰器模式
     * 使用实体装饰类装饰
     * @param args
     */
    public static void main(String[] args) {
        A a = new A();
        B b = new B();
        RedShapeDecorator rda = new RedShapeDecorator(a);
        rda.draw();
        BlueShapeDecorator rdb = new BlueShapeDecorator(b);
        rdb.draw();
    }
}

/**
 * 接口
 */
interface Shape{
    void draw();
}

/**
 * 接口实现类
 */
class A implements Shape{
    @Override
    public void draw() {
        System.out.println("这是A类");
    }
}
class B implements  Shape{
    @Override
    public void draw() {
        System.out.println("这是B类");
    }
}

/**
 * 抽象装饰类
 */
abstract class ShapeDecorator implements Shape{
    protected Shape decoratedShape;

    public ShapeDecorator(Shape decoratedShape) {
        this.decoratedShape = decoratedShape;
    }

    @Override
    public void draw() {
        decoratedShape.draw();
    }
}

/**
 * 第一个实体装饰类
 */
class RedShapeDecorator extends ShapeDecorator{
    public RedShapeDecorator(Shape decoratedShape) {
        super(decoratedShape);
    }

    @Override
    public void draw() {
        super.draw();
        setRedShapeDecorator(decoratedShape);
    }
    private void setRedShapeDecorator(Shape decoratedShape){
        System.out.println("Red");
    }
}
/**
 * 第二个实体装饰类
 */
class BlueShapeDecorator extends ShapeDecorator{
    public BlueShapeDecorator(Shape decoratedShape) {
        super(decoratedShape);
    }
    @Override
    public void draw() {
        super.draw();
        setRedShapeDecorator(decoratedShape);
    }
    private void setRedShapeDecorator(Shape decoratedShape){
        System.out.println("Blue");
    }
}

        在上面的demo中,我们定义了一个普通接口、两个基本的接口实现类、一个抽象装饰类和两个实体装饰类。接口实现类和抽象装饰类都实现了接口的实现方法并重写接口的方法。实体装饰类继承自抽象装饰类,由于抽象装饰类中含有有参构造方法,因此需要使用super关键字声明继承对象。我们通过往实体装饰类对象中传入一个实体类对象,实现了对对象调用的draw()方法的装饰。

        可以看一下这张图,我们可以发现其实在Shape接口上我们定义了一个ShapeDecorator抽象装饰类来装饰,所谓装饰其实也是增加功能。在前面中我们也提及装饰的本质就是使用抽象类继承接口,再使用具体的实现类装饰上功能,这种方式极好的解决了子类爆炸的问题。


总结

        到现在荔枝也梳理了几种设计模式了,总结一下自己的学习方法:主要是根据文档和大佬视频资源讲解梳理的,再自己手敲示例实现一遍,收获和体会确实是挺大的。学习设计模式可能会比较枯燥,大家加油,荔枝要继续前行咯~

今朝已然成为过去,明日依然向往未来!我是小荔枝,在技术成长的路上与你相伴,码文不易,麻烦举起小爪爪点个赞吧哈哈哈~~~ 比心心♥~~~

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

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

相关文章

OpenCV(三十一):形态学操作

​​​​​​1.形态学操作 OpenCV 提供了丰富的函数来进行形态学操作,包括腐蚀、膨胀、开运算、闭运算等。下面介绍一些常用的 OpenCV 形态学操作函数: 腐蚀操作(Erosion): erode(src, dst, kernel, anchor, iteration…

【LeetCode】剑指 Offer <二刷>(6)

目录 题目:剑指 Offer 12. 矩阵中的路径 - 力扣(LeetCode) 题目的接口: 解题思路: 代码: 过啦!!! 题目:剑指 Offer 13. 机器人的运动范围 - 力扣&#…

docker-compose安装mysql

基于docker-compose快速安装mysql 目录 一、目录结构 1、 docker-compose.yml 2、 my.cnf 3、error.log 二、执行安装 三、连接使用 一、目录结构 1、 docker-compose.yml version: 3 services:mysql:image: registry.cn-hangzhou.aliyuncs.com/zhengqing/mysql:5.7 #…

Kubernetes核心概念

Kubernetes是一个工业级的容器编排平台。 Kubernetes核心功能 服务发现和负载均衡。容器的自动装箱,调度(scheduling),按照容器的规格(需要的cpu和内存等)确定一个容器存放到集群中的哪一个机器上。进行自…

创建vue3项目并引用elementui

1.创建vu3项目&#xff1a; 执行命令 npm create vuelatest 2.终端会出现如下选项&#xff0c;不确定的直接enter键进入下一步&#xff1b; 3.然后再执行下方命令&#xff1a; cd <your-project-name> npm install4.安装依赖成功后引入elementui,执行命令&#xff1a…

高压电容器的内部结构是什么样的?

高压电容器的内部结构取决于其具体的设计和用途&#xff0c;但通常包括以下主要组件&#xff1a; 电介质&#xff1a;电介质是高压电容器内部的核心部分。它通常由绝缘材料制成&#xff0c;如聚丙烯薄膜、聚酯薄膜、陶瓷或其他高绝缘性材料。电介质的选择取决于电容器的电压等级…

盘点那些有高级客服分配功能的软件系统

过去&#xff0c;很多企业虽然有服务意识&#xff0c;但并不强烈&#xff0c;现在客户需求以及团队运营的发展推动着企业在客户管理的方式上采用更有效的服务方式来应对实际的变化&#xff0c;尤其是客服行业&#xff0c;所以很多企业在客服的管理和分配上下尽了功夫&#xff0…

Unity的UI管理器

1、代码 public class UIManager {private static UIManager instance new UIManager();public static UIManager Instance > instance;//存储显示着的面板脚本&#xff08;不是面板Gameobject&#xff09;&#xff0c;每显示一个面板就存入字典//隐藏的时候获取字典中对…

【自动化测试】之PO模式介绍及案例

目录 概念 PO三层模式&#xff1a; 1. 构建基础的 BasePage 对象层 2. 构建首页的 Page 层&#xff08;操作层&#xff09; 3.构建业务层 常用断言方法&#xff1a; 4. 构建用例集&#xff0c;执行文件&#xff0c;输出自动化测试报告 测试报告模板 概念 PO&#xff08…

【C++】详解红黑树并模拟实现

前言&#xff1a; 上篇文章我们一起学习了AVL树比模拟实现&#xff0c;我们发现AVL树成功地把时间复杂度降低到了O(logN)。但是同时我们不难发现一个问题&#xff0c;在构建AVL树中我们也付出了不小的代价&#xff0c;频繁的旋转操作导致效率变低。为了解决这个问题&#xff0c…

使用Fastchat部署vicuna大模型

FastChat是一个用于训练、提供服务和评估基于大型语言模型的聊天机器人的开放平台。其核心特点包括&#xff1a; 最先进模型&#xff08;例如 Vicuna&#xff09;的权重、训练代码和评估代码。一个分布式的多模型提供服务系统&#xff0c;配备 Web 用户界面和与 OpenAI 兼容的…

算法通关村第十七关:黄金挑战-跳跃游戏问题

黄金挑战-跳跃游戏问题 1. 跳跃游戏 LeetCode 55 https://leetcode.cn/problems/jump-game/ 思路分析 关键是判断能否到达终点&#xff0c;不用管每一步跳跃到哪里&#xff0c;而是尽可能的跳跃到最远的位置 看最多能覆盖到哪里&#xff0c;只要不断更新能覆盖的距离&#x…

【狂神】Spring5笔记(一)之IOC

目录 首页&#xff1a; 1.Spring 1.1 简介 1.2 优点 2.IOC理论推导 3.IOC本质 4.HelloSpring ERROR 5.IOC创建对象方式 5.1、无参构造 这个是默认的 5.2、有参构造 6.Spring配置说明 6.1、别名 6.2、Bean的配置 6.3、import 7.DL依赖注入环境 7.1 构造器注入 …

[JAVA] byte与int的类型转换案例剖析

总结&#xff1a; ①没有byte的字面值&#xff0c;赋值时需要强制转换类型 ②涉及运算&#xff0c;系统自动进行类型升级&#xff0c;由此用final修饰&#xff0c;代表这是一个不会更改值的常量&#xff0c;通过编译 感受&#xff1a;还是用int吧&#xff0c;自动类型转换太复…

VB:顺序查找

VB&#xff1a;顺序查找 Private Sub Command1_Click()Dim i%, m%Dim x(1 To 10) As SingleFor i 1 To 10x(i) Val(InputBox("请输入"))Next im seqSearch(x, 10)If (m 1) ThenPrint "已找到"ElsePrint "未找到"End If End Sub Function se…

为什么在线客服系统的消息撤回功能是有必要的?

如今在日常工作和沟通中&#xff0c;很多企业都在使用在线客服系统跟客户进行线上交流和协作。然而有时候客服可能会不小心发送错误的消息或包含敏感信息的消息&#xff0c;人们在现实的沟通交流中是不会真实存在“说出去的话还能收回来”的情况&#xff0c;但这是在网络上&…

thinkphp开启定时任务的三种办法(最全)

第一种方法 使用think-cron类库 //composer 安装 composer require yunwuxin/think-cron github文档地址 https://github.com/yunwuxin/think-cron 1.创建任务类 <?php namespace app\task; use yunwuxin\cron\Task; class DemoTask extends Task { public function …

【从入门到起飞】JavaSE—File的使用,构造方法,成员方法

&#x1f38a;专栏【JavaSE】 &#x1f354;喜欢的诗句&#xff1a;更喜岷山千里雪 三军过后尽开颜。 &#x1f386;音乐分享【The truth that you leave】 &#x1f970;欢迎并且感谢大家指出我的问题 文章目录 &#x1f354;File概述&#x1f354;File构造方法⭐根据文件路径…

el-carousel-item轮播一个swiper显示多个卡片数据

效果图&#xff1a; 图片路径均为假地址&#xff0c;需自行替换&#xff0c;1rem100px&#xff0c;可自行转换成px <template><div class"exhibitors page-item-blue-bg"><comItemTitle :titleInfo"titleInfo"> </comItemTitle>…

如何评估需求优先级?

项目的需求来源有很多方面&#xff0c;最终由产品经理整理出来哪些要做&#xff0c;哪些不做。我前面说过需求评审时&#xff0c;要讲清楚这次版本的目的是什么。这些要做的功能就是达成目的的手段。 一般情况下&#xff0c;我们都默认产品或技术总监给需求定优先级。比如优先…