前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >设计模式之工厂模式

设计模式之工厂模式

作者头像
PhoenixZheng
发布2018-08-07 16:20:54
2690
发布2018-08-07 16:20:54
举报

工厂模式作为设计模式的一种在开发中被普遍使用, 其实应该可以说是最经常使用的一种的了。 它的设计思想也是面向接口,如果细分下来,可以分成两种工厂模式 · 工厂方法 - Factory Method · 抽象工厂 - Abstract Factory

今天先把Factory Method说完,因为Abstract Factory的实现跟Factory Method有点相似,为了区分我们还是明天再接着说Abstract Factory。

工厂方法 - Factory Method

刚接触工厂模式的同学可能很容易被名字混淆,工厂这个工厂那个的, 我建议是直接记英文,这样不容易混淆两者的概念和实现,毕竟像 "抽象","方法"这两个词,在文章中出现频率非常高,会产生阅读疲劳。 所以下面我们统一用 Factory Method来说明吧。

举例

讲抽象原理不如直接上例子来的直接。 举个简单的例子,现在需要获得不同的颜色,在没有使用 Factory Method之前,我们的代码看起来可能是这样的

代码语言:javascript
复制
Red red = new Red();
Green green = new Green();
Blue blue = new Blue();
....
red.paint();
green.paint();
blue.paint();

这样没有什么问题,但是代码不够灵活。试想一下如果我们不想要红色了,要修改 Red变成 Yellow,除了实例化的代码,下面还有引用的代码,要改的地方就很多了。 而且从使用者的抽象角度来说,新建对象是一个过程,而不是一个操作,使用者需要关心具体的实例化过程,和实例化的使用,这不是面向对象的好思路。 Factory Method 可以解决这种问题,它把使用者和实例的生产逻辑解耦,从而让使用者只关心实例的使用过程。

Factory Method

对于以上代码,可以看出每个颜色都有共同的方法 paint(),这可以作为一个方法抽象出来,

代码语言:javascript
复制
interface Color {
 void paint();
}

具体的颜色类,这里只举一个类,其他都一样,

代码语言:javascript
复制
public class Red  implements  Color {
    public void paint(){
        System.out.println("paint red");
    }
}
Factory

对于 Factory来说,它负责具体的实例生产过程, 可以这么想象,我不关心 Red实例是怎么来的,我只告诉 Factory我要什么颜色,Factory交给我之后,我只需要调用 paint()接口就行。 所以对于调用者来说,现在的代码可能是这样的,

代码语言:javascript
复制
Color color = ColorFactory.create("red");
....
color.paint();

现在看,是不是在调用者的代码里完全没有出现跟实例生产相关的代码了? 即使后面需要修改具体的颜色,也只需要改掉 create("red") 里的参数就可以。

现在来看看 Factory的代码,

代码语言:javascript
复制
class ColorFactory {
    public static Color create(String color) {
        switch(color) {
            case "red":
                return new Red();
            break;
            ....
        }
    }
}

为了简化代码这里只列出了 Red相关的引用。 so…这就是 Factory Method的精髓所在,通过 Factory把实例的生产过程隐藏起来。

但如果你想深一层,这样的简易版Factory Method也会有问题。 如果后面引入的颜色越来越多, ColorFactory的代码量就会变得越来越长,如果实例化的细节足够复杂,那么 ColorFactory也会变成一个超级上帝类。

Factory Method的进化版

为了避免 ColorFactory被迭代到超级上帝类,可以进一步优化代码。 对于 Factory来说,它只关心一件事情,就是生产实例,所以对于这部分的操作,可以抽象出来作为一个接口 create(),

代码语言:javascript
复制
interface Factory {
    Color create();
}

现在开始,我们就可以抛弃 ColorFactory这个超级上帝类了,对于每种Color,可以有一个Factory,

代码语言:javascript
复制
public class FactoryRed implements Factory {
    Color create() {
        return new Red();
    }
}

public class FactoryBlue implements Factory {
    Color create() {
        return new Blue();
    }
}

这样带来的好处是,每当需要引入新的 Color实现时,只需要新增一个类来实现 Factory接口就可以,而不用再往 ColorFactory里塞代码, 使用者的代码会变成这样,

代码语言:javascript
复制
Factory factory = new FactoryRed();
Color color = factory.create();
color.paint();

相对于简单版的 Factory Method,虽然多了一行代码,但是灵活性不变, 即使需要修改 color实例的具体实现,也只需要修改 FactoryRed成你需要的类就行。

总结

Factory Method是开发的利器, 它可以很好的隔离使用者和生产者的逻辑,从而让使用者能灵活的使用,也只需要关心对象能使用的操作,而不需要关心对象的生产过程。 当你的代码规模变到足够大时,就需要考虑将 Factory进一步解耦,用不同的Factory去生产对应的实例。 如果是在一个团队里开发的话,甚至可以通过这种方式把各个具体的Factory分散到不同的小组去让大家各自实现。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-05-03,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Android每日一讲 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 工厂方法 - Factory Method
  • 举例
  • Factory Method
  • Factory
  • Factory Method的进化版
  • 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档