创建型设计模式01-简单工厂模式

news2024/12/29 8:06:21

✨作者:猫十二懿

❤️‍🔥账号:CSDN 、掘金 、个人博客 、Github

🎉公众号:猫十二懿

这里只是简单的将《大话设计模式【Java溢彩加强版】》的内容简单是复述一下,并加上自己的理解

简单工厂模式

1、引入问题

首先我们看看用Java实现一个简单是计算器程序:

/**
 * @author Shier
 * CreateTime 2023/4/7 16:22
 * 简单的计算器
 */
public class SimpleCalculate {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        System.out.print("请输入数字A:");
        String A = scanner.nextLine();
        System.out.print("请选择进行的操作运行(/、*、-、+):");
        String B = scanner.nextLine();
        System.out.print("请输入数字B:");
        String C = scanner.nextLine();
        double D = 0d;
        if (B.equals("+")) {
            D = Double.parseDouble(A) + Double.parseDouble(C);
        }
        if (B.equals("-")) {
            D = Double.parseDouble(A) - Double.parseDouble(C);
        }
        if (B.equals("*")) {
            D = Double.parseDouble(A) * Double.parseDouble(C);
        }
        if (B.equals("/")) {
            D = Double.parseDouble(A) / Double.parseDouble(C);
        }
        System.out.println("计算结果:" + D);
    }
}

从上面的程序你看出了什么问题了吗?

存在的问题:

  1. 变量名定义不规范(以上程序中使用A、B、C)
  2. 判断分支,进行一次运行,就全部都要去执行,耗时过长,比如做加法,其他的减乘除的判断都要去执行一次,做了三次的无用功。
  3. 除数为0,没有做容错的判断
  4. 大量的使用Double.parseDouble()类型解析

下面我们来改进一下代码:

public class SimpleCalculate {
    public static void main(String[] args) {
        // 代码规范之后
        try {
            Scanner scanner = new Scanner(System.in);
            System.out.print("请输入数字A:");
            Double numberA = Double.parseDouble(scanner.nextLine());
            System.out.print("请选择进行的操作运行(/、*、-、+):");
            String strOperate = scanner.nextLine();
            System.out.print("请输入数字B:");
            Double numberB = Double.parseDouble(scanner.nextLine());
            double result =0d;
            switch (strOperate){
                case "+":
                    result =numberA+numberB;
                    break;
                case "/":
                    result=numberA/numberB;
                    break;
                case "-":
                    result=numberA-numberB;
                    break;
                case "*":
                    result=numberA*numberB;
                    break;
                default:
                    break;
            }
            System.out.println("计算结果:" + result);
        }catch (Exception e)
        {
            System.out.println(e.getMessage());
        }
    }
}

改进之后看起来舒服多了。但是这样就是最好了的吗?

思考:上面的程序,怎样能做到容易维护,容易扩展,又容易复用呢?

2、面向对象编程

一般好的程序都要做到以下四点:

  1. 可维护:修改要修改之处,不需全部改动
  2. 可扩展:有新的功能,只要添加新的功能即可,又不会破坏原来的功能
  3. 可复用:可以多次使用相同的程序
  4. 灵活性好

面向对象的好处:通过封装、继承、多态降低程序的耦合度

不要把所有的功能都写在一起,这样子就会高耦合,难以维护。

所有我们就要将上面的计算器程序中的业务逻辑和界面逻辑分开。也就是计算和控制台显示的内容进行拆分。

怎么进行拆分?

这时候就需要进行业务逻辑的封装,封装成一个方法,在使用到计算的地方,进行调用这个方法即可

3、业务封装

下面就对上面的计算器程序进行封装

测试类中当中修改如下:

// 代码规范之后
try {
    Scanner scanner = new Scanner(System.in);
    System.out.print("请输入数字A:");
    Double numberA = Double.parseDouble(scanner.nextLine());
    System.out.print("请选择进行的操作运行(/、*、-、+):");
    String strOperate = scanner.nextLine();
    System.out.print("请输入数字B:");
    Double numberB = Double.parseDouble(scanner.nextLine());
    double result = Operate.getResult(numberA, numberB, strOperate);
    System.out.println("计算结果:" + result);
} catch (Exception e) {
    System.out.println(e.getMessage());
}

Operate 就是单独的计算类。

如果其他的程序都想使用计算这个功能,只要在调用这个类下的getResult方法,传入对应的参数即可,这样封装的好处就体现出来了。

在其他程序当中都能轻松的使用同样代码,就不是CV工程师了(CV带给我们的只会是劳累的工作),CV会变得高耦合,而如下这样的程序就会降低耦合。

体现可维护、可扩展、可复用的特点。

/**
 * @author Shier
 * CreateTime 2023/4/7 17:21
 */
public class Operate {
    public static double getResult(double numberA, double numberB, String operate) {
        double result = 0d;
        switch (operate) {
            case "+":
                result = numberA + numberB;
                break;
            case "/":
                result = numberA / numberB;
                break;
            case "-":
                result = numberA - numberB;
                break;
            case "*":
                result = numberA * numberB;
                break;
            default:
                break;
        }
        return result;
    }

}

4、紧耦合VS松耦合

在上面的程序中还是存在着一些问题的。比如现在我要增加多一个求平方的功能,虽说是只要在switch 中添加一个case 即可,但是这样会带来紧耦合的作用。也就是说,我只要求算平方这个运算,但是其他的加减乘除都过来一起编译,这样的可能存在影响。(可能有意或无意修改了一些代码,带来的后果是非常大的)

紧耦合:各个模块之间的依赖程度很高,一旦某个模块发生变化会影响到其他模块的运行,导致整个系统变得难以维护和扩展。这种情况下,代码之间的联系比较紧密,如同钢琴里的琴弦一样紧绷。

松耦合:各个模块之间的依赖程度较低,模块之间的关系十分灵活,一个模块的变化不会对其他模块造成太大的影响。这种情况下,代码之间的联系相对较松,如同各自独立演奏的乐器一样。

所以说我们可以使用基础,抽象出一个运算的抽象类,有一个运算结果的方法,然后让每个不同的运算类来继承这个抽象类,从而得到松耦合的作用,更加的灵活。

运算抽象类:

/**
 * @author Shier
 * CreateTime 2023/4/7 17:21
 */
public abstract class Operate {
    public double getResult(double numberA, double numberB) {
        return 0d;
    }
}

加减乘除类:

在这里插入图片描述

image-20230408173743008

5、简单工厂模式

但是根据上面的修改程序之后,我该怎么样去实例化对象呢?下面就来看看这个 简单工厂模式

为了防止以后的需求增加(再增加一个指数运算的功能),我们最好就是单独一个类来作为实例化的对象,这样就相当于一个工厂。

简单工厂模式是一种常用的创建型设计模式,其主要目的是通过一个工厂类来创建不同类对象的实例,而无需直接使用 new 操作符来创建对象。

简单工厂模式包含如下几个角色:

  1. 抽象产品类:定义了工厂创建的产品对象的公共接口。(也就是上面的运算抽象类)
  2. 具体产品类:实现了抽象产品类接口的具体产品,是工厂方法创建的对象。(也就是通过继承抽象类的加减乘除类)
  3. 工厂类:负责创建具体产品的对象,通常包含一个创建产品对象的公共静态方法。(也就是下面的工厂运算类)

在简单工厂模式中,客户端通过调用工厂类的静态方法来获取所需的具体产品对象。工厂类根据客户端传递的参数不同,动态创建不同的具体产品对象,并返回给客户端使用。

与直接使用 new 关键字相比,简单工厂模式使得客户端代码更加灵活,能够随着需求的变化而动态修改创建对象的行为。

简单运算工厂类:

/**
 * @author Shier
 * CreateTime 2023/4/8 17:56
 */
public class OperationFactory {
    public static Operation createOperate(String operate) {
        Operation operation = null;
        switch (operate) {
            case "+":
                operation = new Add();
                break;
            case "/":
                operation = new Div();
                break;
            case "-":
                operation = new Sub();
                break;
            case "*":
                operation = new Mul();
                break;
            default:
                break;
        }
        return operation;
    }
}

只需要输入运算符号,工厂就实例化出合适的对象,通过多态,返回父类的方式实现了计算器的结果

客户端调用:

image-20230408180153263

最后的结构图如下

image-20230408180618239

程序的执行过程如下介绍:

  1. 进入主函数(main),输入numberA,numberB,已经运算符号(这里我选择用加法)
  2. 然后就到简单工厂类进行判断运算符是哪一个运算,这里是加法则进行new Add() 创建这个对象
  3. 然后就进入Add类,继承了Operation类,也要进去Operation类,但是还没有进行调用,此时case + 退出
  4. 通过调用getResult方法,在Add类进行计算,最终将返回结果。

6、UML类图

UML(Unified Modeling Language)图是一种用于软件系统设计和开发的标准化建模语言,是用于构建和可视化面向对象系统的图形表示法。它可以帮助我们更好地理解系统和软件的结构、行为、交互及其功能。UML图通常被用来描述软件的静态结构(如类、对象、接口等)或者动态行为(如用例、时序图等)。UML图包括用例图、类图、时序图、活动图、组件图、部署图等多种类型,可以根据需求选择使用其中的一种或多种类型进行建模。它是一种静态结构图,在统一建模语言(UML)中被用来描述系统的结构,包括类、它们的属性、操作(或方法)以及对象之间的关系。

在Java中,UML图被广泛用于面向对象的程序设计中,常用于展示程序系统的结构和行为。

  • 可以用于描述类与类之间的关系、类的属性和方法等。
  • 类通常用矩形表示,类名在顶部,类的属性和方法分别在中间和底部。
  • 实现类和类之间的关系(如继承关系、关联关系、依赖关系、聚合、组合等)时,多采用UML类图进行表示。

根据下面这个UML类图样例来展开说说Java中类与类的关系:

image-20230408191605911

6.1 介绍UML类图

你可以发现每个矩形框都有三层。

  1. 矩形框:表示每个类。
  2. 第一层:类的名称,若是抽象类,字体则是斜体显示。
  3. 第二层:类的特性,通常就是字段和属性。
  4. 第三层:类的操作,通常是方法或行为。
  5. 第二、三层前面都有一些符号:
    1. +:表示public修饰符
    2. -:表示private修饰符
    3. #:表示protected修饰符

6.1.1 抽象类

表示如下:

image-20230408192220230

6.1.2 接口

接口图与类图的区别:顶端有<<interface>>显示

  • 第一行是接口名称
  • 第二行是接口方法。

接口还有另一种表示方法,俗称棒棒糖表示法,比如图中的唐老鸭类就是实现了’讲人话’的接口

image-20230408192538563

6.2 类与类之间的关系

6.2.1 继承

继承关系用空心三角形+实线来表示

image-20230408192747756

6.2.2 实现接口

实现接口用空心三角形+虚线来表示

image-20230408192842638

6.2.3 关联关系

当一个类 ‘知道’ 另一个类时,可以用关联(association)

关联关系用实线箭头来表示 实际还是实现extends 关键词

image-20230408192948871

你看企鹅和气候两个类,企鹅是很特别的鸟,会游不会飞。更重要的是,它与气候有很大的关联。企鹅需要 ‘知道’ 气候的变化,需要 ‘了解’ 气候规律。

6.2.4 聚合(Aggregation)关系

类与类之间的聚合关系表示为一个类包含了另一个类的实例。这种关系被称为has-a(拥有) 关系,其中一个类是整体,另一个类是部分。聚合表示一种弱的 ‘拥有’ 关系,体现的是A对象可以包含B对象,但B对象不是A对象的一部分。

在UML类图中,聚合关系用带空心菱形的实线来表示,整体指向部分。聚合关系用空心的菱形+实线箭头来表示

一个例子是汽车与车轮之间的聚合关系,其中汽车是整体,而车轮是部分。

image-20230408193108826

大雁与雁群这两个类,大雁是群居动物,每只大雁都属于一个雁群,一个雁群可以有多只大雁。但是当这只大雁脱离了这个雁群,这个雁群依然能很好的继续生活下去,这就说明了大雁对象不是雁群对象的一部分。

6.2.5 合成(Composition)关系

合成(Composition,也有翻译成 ‘组合’ 的)是一种强的 ‘拥有’ 关系,体现了严格的部分和整体的关系,部分和整体的生命周期一样。

合成关系用实心的菱形+实线箭头来表示

  • 合成关系的连线两端还有一个数字 ‘1’ 和数字 ‘2’ ,这些数被称为基数。表明这一端的类可以有几个实例,很显然,一个鸟应该有两只翅膀。
  • 如果一个类可能有无数个实例,则就用 ‘n’ 来表示。
  • 关联关系、聚合关系也可以有基数。

image-20230408193414805

在这里鸟和其翅膀就是合成(组合)关系,因为它们是部分和整体的关系,并且翅膀和鸟的生命周期是相同的。

6.2.6 依赖关系

动物几大特征,比如有新陈代谢,能繁殖。而动物要有生命力,需要氧气、水以及食物等。也就是说,动物依赖于氧气和水。它们之间是依赖关系(Dependency)

用虚线箭头来表示

image-20230408193519307

编程是一门技术,更是一门艺术,不能只满足于写完代码运行结果正确就完事,时常考虑如何让代码更加简练,更加容易维护,容易扩展和复用,只有这样才可以真正得到提高。写出优雅的代码真的是一种很爽的事情。

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

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

相关文章

windows环境下sublime的nodejs插件详细安装图解

前面的话 搜索了好多文档后&#xff0c;才成功地安装了sublime text3的nodejs插件。为了存档&#xff0c;也为了方便有同样需求的朋友&#xff0c;将其安装过程详细记录如下 安装nodejs 虽然nodejs官网提供了node的msi文件&#xff0c;但本人在win7系统下多次尝试&#xff0c;…

在阿里干了6年自动化测试,30岁即将退休的我,告诉你自动化测试工程师有多吃香...

测试人员需要具备自动化测试或者测试开发能力&#xff0c;已经成为测试行业内容的一种显在趋势&#xff0c;而且这种趋势呈放大态势&#xff0c;其发展前景是十分明朗的。 2022这种疫情期间&#xff0c;就业大环境不是很理想&#xff0c;目前呢&#xff0c;企业倾向于招自动化…

Win11下Microsoft Store安装Ubuntu报错解决指南

目录 从Microsoft Store下载Ubuntu下载安装完毕后&#xff0c;打开安装时报错&#xff1a; 一. WslRegisterDistribution failed with error: 0x8007019e 二. WslRegisterDistribution failed with error: 0x800701bc 如果帮到你的话&#xff0c;麻烦点个免费的关注吧bro♥…

测试开发工程师的薪资上限究竟在哪?年薪50W都不是梦...

在说测试开发工程师的薪资待遇之前&#xff0c;咱们要先了解软件测试岗位是用来做什么的&#xff0c;岗位是否重要&#xff0c;只有你知道了这些&#xff0c;才能判断这个岗位是否有价值&#xff01;软件测试是依据需求分析和测试用例&#xff0c;运用手工和自动化的手段来验证…

市场需求升级,cv5200带来新的WiFi传输解决方案,远距离无线通信技术

近年来&#xff0c;随着无线通信技术的不断发展&#xff0c;越来越多的应用需要高速率、远距离、稳定的数据传输。 远距离WiFi传输是指通过WiFi信号实现在远距离范围内的无线通信和数据传输。 为此&#xff0c;cv5200采用了先进的技术&#xff0c;并提供了较远的传输距离、高…

高速高密PCB高级验证技巧(四): 扫除信号线的意外回音

现今电子产品复杂度越趋增加&#xff0c;信号速度越来越快&#xff0c;在信号传输的过程中&#xff0c;如果信号不断反射便会对电子产品的运作造成影响&#xff0c;而这又与阻抗连续性以及阻抗匹配息息相关&#xff1b;而如何避免信号反射&#xff0c;除了在硬件设计时的规划外…

我用过的这5款小工具,你用了其中几款?

有时候一些小工具&#xff0c;能给你带来一些意想不到的效果&#xff0c;我们来看看下面这5款工具&#xff0c;你又用过其中几款呢&#xff1f; 1.文件差异比较工具——WinMerge WinMerge是一款文件和文件夹比较工具,它可以查看文件和文件夹之间的差异,并进行合并。这个工具有…

uCOSii任务管理

uCOSii任务管理 主要用来测试uCOSii“创建任务,挂起任务,恢复任务,发送删除任务请求,删除任务”。 在os_cfg.h中 #define OS_LOWEST_PRIO 63u //设置最低优先级为63,则空闲任务优先级OS_TASK_IDLE_PRIO就等于63 //OS_PRIO_SELF为255,因此OS_LOWEST_PRIO<255 注意&a…

目标检测数据预处理——非宫格与宫格混合拼图(大宽高比图片)

之前一直用的是宫格的正方形拼图&#xff0c;但比如对“人”框的截图是这种高宽高比的长方形图片&#xff0c;按照最大边resize最小边等比例缩放后放入宫格中对造成最小边resize太多&#xff0c;整体图片缩小很多。所以本片专门针对高宽高比的图片拼图进行编辑。 本篇的拼图方式…

一些好用的软件推荐给你

软件一&#xff1a;nTrun nTrun 是一款非常实用的快速启动工具&#xff0c;它可以帮助用户快速启动各种常用的应用程序、网站和文件。此外&#xff0c;nTrun 还具有以下强大的功能&#xff1a; 自定义快捷键&#xff1a;用户可以根据自己的需求为每个应用程序、网站或文件设置…

Mysql链接工具

众所周知为了可以更好的操作 Mysql 数据库&#xff0c;我们都会采用远程连接工具的方式连接 Mysql 数据库&#xff0c;使用远程连接工具连接的好处在于&#xff1a; 方便远程访问&#xff1a;如果你需要在外部网络环境中访问 MySQL 数据库&#xff0c;使用远程连接工具可以方便…

《人生十二法则》- 解决人生80%不如意

法则一获胜的龙虾从不低头&#xff1a;笔直站 立&#xff0c;昂首挺胸。 法则二像照顾生病的宠物一样关心自 己&#xff1a;待己如助人。 法则三放弃损友&#xff1a;与真心希望你好的人 做朋友。 法则四战胜内心的批评家&#xff1a;和昨天的自 己比&#xff0c;别和今天的…

微软官方Microsoft Remote Desktop for Mac

microsoft-remote-desktop-for-mac 时候还是需要用到windows系统上的数据或者软件&#xff0c;除了使用第三方开发商的远程桌面工具外&#xff0c;微软公司也提供了Mac版&#xff08;iMac和MacBook&#xff09;的远程桌面软件&#xff08;Microsoft Remote Desktop&#xff09…

大数据Doris(二十五):Doris数据Binlog Load导入方式介绍

文章目录 Doris数据Binlog Load导入方式介绍 一、基本原理 二、Canal原理及配置 1、Canal同步MySQL数据原理 2、开启MySQL binlog 3、Canal配置及启动 三、Doris同步MySQL数据案例 1、MySQL中创建源表 2、Doris中创建目标表 3、创建同步作业 四、注意事项 1、关于配…

关于我用python下载两千四百四十四章保存txt这件事。。。

前言 大家早好、午好、晚好吖 ❤ ~欢迎光临本文章 女同事最近迷上了一本书 但她又不想下载软件&#xff0c;就想要我给你下载成txt慢慢看 一看章节&#xff0c;两千四百四十四章&#xff0c;这我能答应嘛&#xff1f; 面对美女小姐姐的请求&#xff0c;我当场表示&#xff1…

看ChatGPT是如何教我爬取上千家上市公司的股票代码

现在有一个这样的需求&#xff0c;要爬取雪球网上A股的股票名称、代码和总市值这些信息并把它保存到execl表格中。对于一个新手想学习爬虫&#xff0c;如何通过chatGPT来完成这个任务呢&#xff1f; 首先&#xff0c;我们把自己的需求详细的描述向ChatGPT提问&#xff0c;问题…

数据库可视化神器,你在用哪一款呢

唠嗑部分 在我们日常开发中&#xff0c;作为开发者&#xff0c;与数据库是肯定要打交道的&#xff0c;比如MySQL&#xff0c;Oracle、sqlserver… 那么数据库可视化工具&#xff0c;你用什么呢&#xff1f;小白今天将常用地几款工具列一下&#xff0c;各位小伙伴如有喜欢的自…

亚马逊开放个人卖家验证入口?亚马逊卖家验证到底怎么搞?

亚马逊卖家账户的安全对于所有卖家来说都非常重要。如果卖家想要在亚马逊上长期稳定地发展&#xff0c;赚取更多的钱并推出更多热卖产品&#xff0c;就必须确保他们的亚马逊卖家账户安全&#xff0c;特别是一直存在的亚马逊账户验证问题。 近期&#xff0c;根据亚马逊官方披露的…

【VPX302】基于3U VPX总线架构的高性能数据预处理平台/XCKU115

板卡概述 VPX302是一款基于3U VPX总线架构的高性能数据预处理FMC载板&#xff0c;板卡具有1个FMC&#xff08;HPC&#xff09;接口&#xff0c;1个X8 GTH背板互联接口&#xff0c;可以实现1路PCIe x8&#xff1b;具有4路SRIO X4。板卡采用Xilinx的高性能Kintex UltraScale系列F…

简单实现远程访问Linux SVN服务

文章目录 前言1. Ubuntu安装SVN服务2. 修改配置文件2.1 修改svnserve.conf文件2.2 修改passwd文件2.3 修改authz文件 3. 启动svn服务4. 内网穿透4.1 安装cpolar内网穿透4.2 创建隧道映射本地端口 5. 测试公网访问6. 配置固定公网TCP端口地址6.1 保留一个固定的公网TCP端口地址6…