设计模式——简单工厂模式

计算器的例子

以面向过程的方式做一个简单实现

在本例子中,将通过一个简单的实现方式,做一个简单的加减乘除的计算器。

CalcSimpleImpl类

public class CalcSimpleImpl {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        System.out.print("请输入第一个数:");
        int firstInput = Integer.valueOf(scanner.nextLine());

        System.out.print("请输入操作符:");
        String operator = scanner.nextLine();

        System.out.print("请输入第二个数:");
        int secondInput = Integer.valueOf(scanner.nextLine());

        String result = "";
        switch (operator) {
            case "+":
                result = firstInput + secondInput + "";
                break;
            case "-":
                result = firstInput - secondInput + "";
                break;
            case "*":
                result = firstInput * secondInput + "";
                break;
            case "/":
                result = firstInput / secondInput + "";
                break;
            default:
                break;
        }

        System.out.println("计算结果:" + result);
    }
}

在上述代码中,很容易看出一些问题。

  1. 所有的输入没有做相关校验,这里我们不多说,本文的重点不在这里。

  2. 做除法的时候没有做被除数为0 的判断

  3. 代码完全面向过程操作,可维护性极低,如果说以后加需求,需要加上一个算平方跟的计算方式,那么就需要添加switch里面的case,需要定位到具体算法的位置才可以实现。


以面向对象的方式修改实现

下面开始针对上述问题进行修改(第1条暂不修改,本文针对第2、3条进行修改)

首先,我们需要一个面向对象的思维进行修改,将具体的计算方法进行封装,对调用方(main方法)进行隐藏,仅仅暴露出一个getResult方法。

Operation类

public class Operation {
    
    private String type;

    public Operation(String type) {
        this.type = type;
    }

    public String getResult(int firstInput, int secondInput) {
        String result = "";
        switch (this.type) {
            case "+":
                result = firstInput + secondInput + "";
                break;
            case "-":
                result = firstInput - secondInput + "";
                break;
            case "*":
                result = firstInput * secondInput + "";
                break;
            case "/":
                result = firstInput / secondInput + "";
                break;
            default:
                break;
        }
        return result;
    }
}

CalculatorOop类

public class CalculatorOop {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        System.out.print("请输入第一个数:");
        int firstInput = Integer.valueOf(scanner.nextLine());

        System.out.print("请输入操作符:");
        String operator = scanner.nextLine();

        System.out.print("请输入第二个数:");
        int secondInput = Integer.valueOf(scanner.nextLine());

        Operation operation = new Operation(operator);
        System.out.println("计算结果:" + operation.getResult(firstInput, secondInput));
    }
}

经过上面的修改之后,我们的代码具备了面向对象的部分特性。
但是仍然不能算是一个优秀的代码,当我们需要新增某种操作符的运算时,我们需要对Operation类中的switch进行修改,存在更改到其他运算符的实现的风险。


使用简单工厂模式进行修改

针对简单实现中的第3条问题,在本例中,各个操作符算法之间是相互独立的。
而具体的运算算法的使用,由使用方的传入值所决定,我们可以进行如下的修改。

定义一个Operator操作接口,接口中有一个getResult方法,每一种运算操作都各自实现它。
而用户在使用时,只需要通过该接口进行操作,无需关心内部如何实现。
再定义一个OperatorFactory类,根据不同的输入值,获取不同的Operator操作实例。

OperatorFactory类

public class OperatorFactory {

    public static Operator getOperator(String type) {
        Operator operator = null;
        switch (type) {
            case "+":
                operator = new OperatorAdd();
                break;
            case "-":
                operator = new OperatorSub();
                break;
            case "*":
                operator = new OperatorMulti();
                break;
            case "/":
                operator = new OperatorDiv();
                break;
            default:
                break;
        }

        // 可能会存在为没有定义的操作符,使得operator为null,但是不在本次文章的重点中

        return operator;
    }
}

Operator接口

public interface Operator {

    String getResult(int firstInput, int secondInput);
}

四种运算操作类

public class OperatorAdd implements Operator {
    public String getResult(int firstInput, int secondInput) {
        return firstInput + secondInput + "";
    }
}
public class OperatorSub implements Operator {
    public String getResult(int firstInput, int secondInput) {
        return firstInput - secondInput + "";
    }

}
public class OperatorMulti implements Operator {
    public String getResult(int firstInput, int secondInput) {
        return firstInput * secondInput + "";
    }
}
public class OperatorMulti implements Operator {
    public String getResult(int firstInput, int secondInput) {
        return firstInput * secondInput + "";
    }
}

CalculatorMain类

public class CalculatorMain {

    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);

        System.out.print("请输入第一个数:");
        int firstInput = Integer.valueOf(scanner.nextLine());

        System.out.print("请输入操作符:");
        String operatorInput = scanner.nextLine();

        System.out.print("请输入第二个数:");
        int secondInput = Integer.valueOf(scanner.nextLine());

        Operator operator = OperatorFactory.getOperator(operatorInput);
        System.out.println("计算结果:" + operator.getResult(firstInput, secondInput));
    }
}

总结

好,至此一个简单工厂模式的计算器实现方法就编写完成了。
当我们需要新增一个新的操作符算法时,只需要新增加一个类,并在操作符工厂中,添加上对应的返回类型即可,将具体算法对客户端隐藏,也对工厂类隐藏,具体的运算算法只对各个操作符开放。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,519评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,842评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,544评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,742评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,646评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,027评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,513评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,169评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,324评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,268评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,299评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,996评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,591评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,667评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,911评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,288评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,871评论 2 341

推荐阅读更多精彩内容