Java 设计模式 -- 工厂方法模式

简单工厂模式(Simple Factory Pattern)

定义一个工厂类,可以根据参数的不同返回不同类的实例,被创建的实例通常拥有共同的父类,因为简单工厂模式中用于创建实例的方法是静态方法,因此简单工厂模式又被称为静态工厂方法。
假如有一家汽车厂,它可以生产很多类型的汽车,那么这个类我们可以设计为以下格式。

public class CarFactory {

    public static final int AUDI = 0;
    public static final int BMW = 1;
    private int type;

    public CarFactory(int type) {
        this.type = type;
    }

    public void createCar() {
        if (type == AUDI) {

            // create AUDI
        } else if (type == BMW) {

            // create BMW
        }
    }
}

这个时候可以分析一下,这个类有以下缺点

1.代码过于冗长,阅读难度,维护难度,测试难度增大。

2.CarFactory 类的职责太重,因为它不仅包含 createAudi 而且还有 createBMW。

3.包含了太多的 if...else 判断语句,一旦要类的需求发生变动,需要改变CarFactory 源代码,违反了开闭原则。

4.客户端通过 new 操作创建和实例化CarFactory,也就是说二者耦合度过高,因为对象的创建和使用无法分离。


那么使用简单工厂模式能做些什么呢?我们可以将 对象的创建和使用分离,也就是在客户端与生产之间搭建一个中间类,这个类通过控制产品类的创建返回给客户端一个期望的值。

首先,我们需要为所有产品创建一个接口或者抽象类,那么什么时候使用抽象类什么时候使用接口呢?当所有的产品类中有共同的方法时,应该使用abstract类,否则使用接口。举例来说,如果所有汽车都需要盖上厂家自己的商标,那么所有类型的车都应该盖同一个标志,这就是一个共同的方法。

public abstract class AbstractCar {

    public void setIcon(){

    }

    public abstract void show();
}

如果没有共同方法,使用interface 就足够了

  public interface InterfaceCar {

        public void show();
}

下面使用简单工厂模式再次实现上面那个汽车厂的功能。

  public class Audi implements InterfaceCar{

    @Override
    public void show() {

        // Create Audi

    }

}

public class BMW implements InterfaceCar{

    @Override
    public void show() {
        // Create BMW
    }

}

public class CarFactory {

    public static final int AUDI = 0;
    public static final int BMW = 1;

    public static InterfaceCar createCar(int type){

        if (type == AUDI) {

            return new Audi();

        } else if (type == BMW) {

            return new BMW();

        } else {

            return null;
        }
    }
}

可以看出简单工厂模式有以下优点:

1.客户端可以免除创建一个工厂类的实例,并且直接通过工厂类的静态方法获得自己想要的产品,这样就降低了客户端与工厂类的耦合度,实现了对象的创建和使用分离。

2.CarFactory 只通过调用产品实现类的方法去获得一个实例,并没有自己来实现它,也就是说现在这个类只有一个功能,那就是管理创建满足客户端的需求,符合单一职责原则。

当然,这个时候我们需要创建具体的Car 对象时,还是需要到客户端中去修改我们需要的类型,每次修改客户端都需要重新编译,而且每次都是去修改主函数中的代码,这严重违背了开闭原则,有没有一种办法能够让我们既不修改客户端代码,又可以适应我们变化的需求呢?当然是可以的,但是需要借助于 xml。

添加一个 cartype.xml 文件,内容如下

<cartype>
    <type>Audi</type>
</cartype>

这个type 就是我们需要的那个车型,以后如果需要变化,只需要修改这个值即可,不再需要去修改主函数中的代码。当然,首先我们需要再定义一个函数去解析这个type 的值。这个函数的实现如下图所示

 public class XMLUtil {


    public static int getType(){    
        try {
            DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
            DocumentBuilder builder = factory.newDocumentBuilder();
            Document document = builder.parse(new File("cartype.xml"));

            NodeList list = document.getElementsByTagName("type");
            Node node = list.item(0).getFirstChild();

            String carType = node.getNodeValue().trim();

            if(carType.equals("Audi")){
                return CarFactory.AUDI;
            }
            if(carType.equals("BMW")){
                return CarFactory.BMW;
            }

        } catch (ParserConfigurationException e) {
            e.printStackTrace();

        } catch (SAXException e) {
            e.printStackTrace();

        } catch (IOException e) {
            e.printStackTrace();

        }
        return -1;
    }
}

注意,如果你使用的编译器是Eclipse,而且恰巧的是你在指定的包下创建了这个xml文件,那么这个时候会出现查找不到指定xml文件的错误,因为Eclipse中创建的xml文件是和我们 的源代码在同一目录的,解决的方法是复制一份xml文件到项目的根目录下。

好了,这个时候简单工厂方法已经介绍完毕了,但是有一个突出的问题摆在了面前,工厂类中为什么还是存在if...else 判断语句,如果要添加车型,不还是要进工厂类添加if...else 语句吗?这不还是要违背开闭原则吗?解决这个问题最好的办法就是摒弃简单工厂模式,使用工厂方法模式。那么简单工厂模式存在的意义是什么?实际上简单工厂模式在只需要创建很少的对象而且基本上不会出现变动的情况下是比较合理的选择,因为这个时候if...else 语句比较少,逻辑还是比较清楚的,由于很少会出现变动,这个时候使用简单工厂模式比工厂方法模式要高效的多,原因在下面进行解释。

工厂方法模式(Factory Method Pattern)

在简单工厂模式中,通过一个InterfaceCar接口,将各类Car 的创建从 CarFactory类中分离出来,从一定程度上减轻了CarFactory 的负担,但是这么多类型的Car 都由CarFactory 负责创建,任务终究还是比较重的,所以我们可以用类似的方法,定义一个InterfaceFactory类,让一个特定的 Car 由 一个特定的 Factory 去创建,这样设计的话不仅符合单一职责原则,而且还有一个相当清晰的逻辑关系。

附注:不要问我工厂方法模式的定义是什么,因为我也不知道。个人认为只要了解这个思想就可以,不必去硬记定义,毫无意义。

将上述的CarFactory类换成接下来三个类

 public interface InterfaceFactory {    
    public InterfaceCar createCar();
}

public class AudiFactory implements InterfaceFactory{

    @Override
    public InterfaceCar createCar() {
        return new Audi();
    }

}

public class BMWFactory implements InterfaceFactory{

    @Override
    public InterfaceCar createCar() {
        return new BMW();
    }

}

public class Main {

    public static void main(String[] args) {

        InterfaceFactory factory = new AudiFactory();

        InterfaceCar car;
        car = factory.createCar();
        car.show();
    }
}

这样我们就可以通过在客户端指定工厂去生产指定车型的汽车了,不过和简单工厂方法模式有一个类似的问题,我们能不能每次不修改主函数中的代码,就能变更我们的需求,比如说,我现在想要一辆 BMW,主函数中是创建一个Audi,该怎么做才能既不修改源代码,又能满足我的需求,用类似的方法。

首先介绍一个概念,Java 反射机制,也就是指在程序运行时获取已知名称的类或者已有对象的相关信息的一种机制。

接下来看一下 xml文件中的内容

<?xml version="1.0" encoding="UTF-8"?>

<factory>
    <type>AudiFactory</type>
</factory>

这个时候的XML解析类如下面代码所示

 public class XMLUtil {


    @SuppressWarnings("rawtypes")
    public static Object getType(){    
        try {
            DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
            DocumentBuilder builder = factory.newDocumentBuilder();
            Document document = builder.parse(new File("factory.xml"));

            NodeList list = document.getElementsByTagName("type");
            Node node = list.item(0).getFirstChild();

            String mFactory = "com.testfactory."+node.getNodeValue();


            Class mClass = Class.forName(mFactory);
            Object mObject = mClass.newInstance();
            return mObject;    

        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }
}

附注:一定要注意,使用 Class.forname(name:String) 方法时,这个地方传进来的类名一定要是完整的路径,也就是包名+类名,否则就会抛出ClassNotFoundException异常。

下面只需要在主函数中这样定义,就可以一劳永逸了,以后改变需求只需要到xml文件中去修改即可

 public class Main {

    public static void main(String[] args) {

        InterfaceFactory factory;
        InterfaceCar car;

        factory = (InterfaceFactory)XMLUtil.getType();

        car = factory.createCar();
        car.show();
    }
}

工厂方法模式的优点很显然,指定工厂生产指定产品,符合单一职责原则,增加产品类型的时候也只需要增添相应的工厂和产品实现类,无需对已有的代码进行改动,符合开闭原则。既然工厂方法模式这么好,为什么还存在上述的简单工厂模式呢?因为工厂方法模式有一个致命的缺点,就是每增添一个新的产品,就要增加两个类,即工厂类和产品类,也就是说,会有太多的类参与编译运行,会给系统带来不乐观的开销。

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

推荐阅读更多精彩内容