设计模式 - 简单工厂模式

文章目录


前言

大家好,今天给大家介绍一下23种常见设计模式中的一种 - 工厂模式


1 . 问题引入

请用C++、Java、C#或 VB.NET任意一种面向对象语言实现一个计算器控制台程序,要求输入两个数和运算符 号,得到结果。

下面的代码实现默认认为两个操作数为Integer类型, 为了简单起见, 不引入泛型

2 . Version1

public class Version1 {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        System.out.print("请输入数字A: ");
        int A = sc.nextInt();
        System.out.print("请输入你要执行的操作(+,-,*,/): ");
        String str = sc.next();
        System.out.print("请输入数字B: ");
        int B = sc.nextInt();

        if("+".equals(str)){
            System.out.println(A+B);
        }else if("-".equals(str)){
            System.out.println(A-B);
        }else if("*".equals(str)){
            System.out.println(A*B);
        }else if("/".equals(str)){
            System.out.println(A/B);
        }
    }
}

代码问题分析

上述代码对于初学者来说能写出来我想再正常不过了,哈哈, 我们来针对上述三点改进一下,命名还是按照A,B吧,毕竟又不是真的计算器。

3 . Version2 

public class Version2 {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        try{
            System.out.print("请输入数字A: ");
            int A = sc.nextInt();
            System.out.print("请输入你要执行的操作(+,-,*,/): ");
            String str = sc.next();
            System.out.print("请输入数字B: ");
            int B = sc.nextInt();
            int ret = 0;

            switch (str){
                case "+":
                    ret = A+B;
                    break;
                case "-":
                    ret = A-B;
                    break;
                case "*":
                    ret = A*B;
                    break;
                case "/":
                    if(B!=0) {
                        ret = A/B;
                        break;
                    }
                    else throw new RuntimeException("除数为0");
                default:
                    throw new RuntimeException("没有该运算符!");
            }
            System.out.println("结果为: "+ret);
        }catch(Exception e){
            System.out.print("您的输入有误!: ");
            e.printStackTrace();
        }
    }
}

现在在看这段代码,是不是感觉没啥毛病了!  如果你没有学过面向对象编程,我没什么可说的,但是Java是面向对象编程的语言啊!  面向对象的三大特征是啥?

  1. 封装:封装是指将数据和行为(方法)封装在一个类中,并对外部隐藏对象的内部实现细节,只提供公共的访问方式。这样可以保护数据不被直接访问和修改,提高代码的安全性和可维护性。

  2. 继承:继承是指一个类(子类)可以继承另一个类(父类)的属性和方法,子类可以复用父类的代码,并且可以在不改变父类的情况下进行扩展和修改。通过继承可以建立类之间的层次关系,提高代码的可复用性和扩展性。

  3. 多态:多态是指同一个方法在不同的对象上有不同的行为表现。在面向对象编程中,多态可以通过继承和接口实现。多态性可以提高代码的灵活性和可扩展性,使代码更易于维护和扩展。

一个都没用到,哈哈,这肯定是不行的,还得再改进一下!

那么如何改进呢?  先来考虑封装, 是不是可以把操作逻辑和业务逻辑单独封装为一个类?

4. Version3

package FactoryModel.Option3;

import java.util.Scanner;

/**
 * 业务代码 和 计算代码实现分离! - 封装!
 */
public class Version3 {
    public static void main(String[] args) {
        try(Scanner sc = new Scanner(System.in)){
            System.out.print("请输入数字A: ");
            int A = sc.nextInt();
            System.out.print("请输入你要执行的操作(+,-,*,/): ");
            String str = sc.next();
            System.out.print("请输入数字B: ");
            int B = sc.nextInt();
            if(B!=0) System.out.println(Operation.getResult(A,B,str));
            else throw new RuntimeException();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

/**
 * 孺鸟可教也,写得不错,这样就完全把业务和界面分离了。
 * 如果你现在要我写一个Windows应用程
 * 序的计算器,我就可以复用这个运算类(Operation)了
 */
class Operation{
    public static int getResult(int A,int B, String oper){
        int ret = 0;
        switch (oper){
            case "+":
                ret = A+B;
                break;
            case "-":
                ret = A-B;
                break;
            case "*":
                ret = A*B;
                break;
            case "/":
                ret = A/B;
        }
        return ret;
    }
}
现在如果我希望增加一个开根(sqrt)运算,你如何改?
那只需要改Operation类就行了,在switch中加一个分支就行了
问题是你要加一个平方根运算,却需要让加减乘除的运算都得来参与编译,如果你一不小心,把加法运算改成了减法,这岂不是大大的糟糕。况且改代码这件事情不符合 开放 - 封闭原则

开放 - 封闭原则(Open-Closed Principle)是面向对象设计原则之一,提出者是Bertrand Meyer。该原则指出一个软件实体(类、模块、函数等)应该对扩展开放,对修改封闭。换句话说,一个软件实体应该通过扩展来实现新功能,而不是通过修改已有的代码来实现。

具体来说,开放 - 封闭原则要求在系统需要变化时,应该通过添加新的代码来扩展功能,而不是修改已有的代码。这样可以保持系统的稳定性,减少对已有代码的影响,同时也更容易实现代码的复用和维护。

遵循开放 - 封闭原则可以使代码更加灵活、可扩展和可维护,同时也有利于降低系统的耦合度,提高代码的可复用性。这一原则在面向对象设计中扮演着重要的角色,帮助我们设计出更加稳定和易扩展的软件系统。

这个时候就该,继承登场了!

5 . Version4

public abstract class Operation {
    protected Integer A;
    protected Integer B;

    public Integer getA() {
        return A;
    }

    public void setA(Integer a) {
        A = a;
    }

    public Integer getB() {
        return B;
    }

    public void setB(Integer b) {
        B = b;
    }

    public abstract Integer getResult();
}
class OperationAdd extends Operation{
    @Override
    public Integer getResult() {
        return A+B;
    }
}

class OperationDiv extends Operation{
    @Override
    public Integer getResult() {
        return A/B;
    }
}


class OperationMul extends Operation{
    @Override
    public Integer getResult() {
        return A*B;
    }
}

class OperationSub extends Operation{
    @Override
    public Integer getResult() {
        return A-B;
    }
}

这个时候如果我们再额外添加运算是不是就变得很简单了,只需要增加一个类,继承Operation即可!

6 . 简单工厂模式

简单工厂模式(Simple Factory Pattern)是一种创建型设计模式,属于工厂模式的一种。在简单工厂模式中,有一个工厂类负责根据客户端的需求创建相应的产品对象,而客户端无需知道具体产品的创建细节,只需要通过工厂类来获取所需的产品对象。

/**
 * 简单工厂模式
 */

public class OperationFactory {
    public static Operation createOperate(String oper){
        Operation operation = null;
        try{
            switch (oper){
                case "+":
                    operation = new OperationAdd(); // 多态
                    break;
                case "-":
                    operation = new OperationSub(); // 多态
                    break;
                case "*":
                    operation = new OperationMul(); // 多态
                    break;
                case "/":
                    operation = new OperationDiv(); // 多态
                    break;
                default:
                    throw new RuntimeException();
            }
        }catch(Exception e){
            e.printStackTrace();
        }
        return operation;
    }

    public static void main(String[] args) {
        Operation operate = OperationFactory.createOperate("+");
        operate.setA(10);
        operate.setB(20);
        System.out.println(operate.getResult());
    }
}

总结

以上就是这篇博客的主要内容了,大家多多理解,下一篇博客见!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2. Java基本语法

文章目录 2. Java基本语法2.1 关键字保留字2.1.1 关键字2.1.2 保留字2.1.3 标识符2.1.4 Java中的名称命名规范 2.2 变量2.2.1 分类2.2.2 整型变量2.2.3 浮点型2.2.4 字符型 char2.2.5 Unicode编码2.2.6 UTF-82.2.7 boolean类型 2.3 基本数据类型转换2.3.1 自动类型转换2.2.2 强…

报错:torch.distributed.elastic.multiprocessing.errors.ChildFailedError:

错误: torch.distributed.elastic.multiprocessing.errors.ChildFailedError: 这个主要是torch的gpu版本和cuda不适配 我的nvcc -V是11.8 torch使用的: pip install torch2.0.1 torchvision0.15.2 torchaudio2.0.2 --index-url https://download.pyt…

Day47:WEB攻防-PHP应用文件上传函数缺陷条件竞争二次渲染黑白名单JS绕过

目录 文件上传与测试环境安装 1、前端 JS 2、.htaccess(apache独有的配置文件) 3、MIME类型 4、文件头判断 5、黑名单-过滤不严 6、黑名单-过滤不严 7、低版本GET-%00截断 8、低版本POST-%00截断 9、黑名单-过滤不严 10、逻辑不严-条件竞争 11、二次渲染 12、函数…

Leetcoder Day43| 单调栈2

503.下一个更大元素II 给定一个循环数组(最后一个元素的下一个元素是数组的第一个元素),输出每个元素的下一个更大元素。数字 x 的下一个更大的元素是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该…

2024最新Guitar Pro 8.1中文版永久许可证激活

Guitar Pro是一款非常受欢迎的音乐制作软件,它可以帮助用户创建和编辑各种音乐曲谱。从其诞生以来就送专门为了编写吉他谱而研发迭代的。 尽管这款产品可能已经成为全球最受欢迎的吉他打谱软件,在编写吉他六线谱和乐队总谱中始终处于行业领先地位&#x…

大话设计模式之原型模式

原型模式(Prototype Pattern)是一种创建型设计模式,它用于创建对象的复制,同时又能保持对象的封装。原型模式通过复制现有对象的方式来创建新的对象,而无需知道具体创建过程的细节。 在原型模式中,通常会有…

Excel·VBA数组分组问题

看到一个帖子《excel吧-数据分组问题》,对一组数据分成4组,使每组的和值相近 目录 代码思路1,分组形式、可分组数代码1代码2代码2举例 2,数组所有分组形式举例 这个问题可以转化为2步:第1步,获取一组数据…

【大数据运维】minio 常见shell操作

文章目录 1. 安装2. 入门操作3. 命令帮助 1. 安装 下载 https://dl.min.io/client/mc/release/linux-amd64/ 赋权与使用 cp mc /usr/bin && chmod x /usr/bin/mc ./mc --help 2. 入门操作 # 添加minio到mc mc config host add minio_alias_name endpoint_adress …

SpringBoot动态数据源实现

一、背景 一个应用难免需要连接多个数据库,像我们系统起码连接了5个以上数据库,AWS RDS主库,ECS自搭MySQL从库,工厂系统三个SQLServer数据库,在线网站MySQL数据库,记得很早以前是用SessionFactory配置&…

Java中有哪些容器(集合类)?

Java中的集合类主要由Collection和Map这两个接口派生而出,其中Collection接口又派生出三个子接 口,分别是Set、List、Queue。所有的Java集合类,都是Set、List、Queue、Map这四个接口的实现 类,这四个接口将集合分成了四大类&#…

C语言--编译和链接

1.翻译环境 计算机能够执行二进制指令,我们的电脑不会直接执行C语言代码,编译器把代码转换成二进制的指令; 我们在VS上面写下printf("hello world");这行代码的时候,经过翻译环境,生成可执行的exe文件&…

WebGIS概述

1.地图组成 底图(Map): 所有信息的载体 图层(Layer):将不同地理信息分类形成的一个集合 要素(Feature):表示不同的地物 几何(Geometry): 信息的数据模型和抽象 2.地图容器Container 即在准备阶段所创建的指定了id的div对象,这个div将作为承载所有图层、点标记、矢量…

分布式部署LNMP+WordPress

需要四台虚拟机,实际上,我们只需要操作三台 一个数据库,一个nginx,一个php,还需要准备一个软件包wordpress-4.7.3-zh_C 首先配置nginx的服务环境 [rootnginx ~]# vi /usr/local/nginx/conf/nginx.conf 修改文件中的loc…

2024软件设计师备考讲义——(4)

知识产权和标准化 一、知识产权 1.特性 无体性专有性地域性时间性 2.保护期限 公民作品 署名权、修改权、保护作品完整权【没有限制】发表权、使用权、获得报酬权【终身及死亡后第50年12月31日】单位作品 发表权、使用权、获得报酬权【首次发表后到第50年12月31日】公民软件…

【Linux】nmcli命令详解(文末送书)

目录 一、概述 二、常用参数使用 2.1 nmcli networking 1.显示NM是否接管网络 2.查看网络连接状态 3.开/关网络连接 2.2 general ​编辑 1.显示系统网络状态 2.显示主机名 3.更改主机名 2.3 nmcli connection ​编辑1.显示所有网络连接 2.显示某个网卡的详细信息…

修改mysql数据库默认字符集

查看系统版本,数据库版本 前提你必须已经安装好了mysql。 参考:https://blog.csdn.net/qq_50247813/article/details/137137915 查看mysql的默认字符集 show variables like %char%; 查看数据库默认字符集 SELECT collation_database; 查看数据库默认…

携手伙伴 共赢智改数转 锐捷网络企业行业合作伙伴大会圆满举行

3月22日,锐捷网络2024全国企业行业合作伙伴大会在福州成功举行。大会以“追光而遇,沐光同行”为主题,吸引了来自全国各地的合作伙伴齐聚“有福之州”,共同探讨企业数智化转型新机遇和新方向。 会上,锐捷网络渠道客户系统部总经理王刚为此次合作伙伴大会开幕致辞。王刚对所有到场…

13 Games101 - 笔记 - 光线追踪(Whitted-Style光线追踪原理详解及实现细节)

13 光线追踪(Whitted-Style光线追踪原理详解及实现细节) 引入光线追踪的原因 光栅化的缺点:不能很好的处理全局光照。(因为Blinn-Phong这种局部模型无法处理全局效果!) 光栅化:快 real-time 质量低光线追…

亚马逊、Shine新品如何快速引爆流量?自养号测评实用技巧助你成功

在当今电子商务的浪潮中,亚马逊凭借其卓越的运营模式和庞大的用户基础,已然成为全球在线零售领域的佼佼者。然而,面对平台上数以亿计的商品和激烈的竞争环境,新品要想快速吸引流量并脱颖而出,并非易事。本文旨在深入探…

Flink-CDC 无法增量抽取SQLServer数据

1.问题 因部署在WindowsServer服务器SQLServer发生过期后重启,Flink-CDC同步进行作业重启,启动后无报错信息,数据正常抽取。但是观察几天后发现当天数据计算指标无法展示 2.定位 因为没用进行任何修改,故初步判断不是因Flink-C…