设计模式 - 工厂方法模式 (2/23)

简单工厂

简单工厂也叫静态工厂

  1. 简单工厂是属于创建型模式。简单工厂是由一个工厂对象决定创建出哪一种产品类的实例。简单工厂是工厂模式家族中最简单的实用模式
  2. 简单工厂:定义了一个创建对象的类,由这个类来封装实例化对象的行为

在开发中,当我们会用到大量的创建某种、某类、或者某批对象时,就会使用到工厂模式。

简单的例子

生鲜

public abstract class Fresh {

    protected String name;

    public abstract void prepare();

    public void washing() {
        System.out.println(name + "正在清洗");
    }

    public void cut() {
        System.out.println(name + "正在分切");
    }

    public void box() {
        System.out.println(name + "正在打包");
    }

}

黄瓜

public class Cucumber extends Fresh {

    public Cucumber() {
        super.name = "黄瓜";
    }

    @Override
    public void prepare() {
        System.out.println(super.name + "正在准备");
    }
}

西蓝花

public class Broccoli extends Fresh {

    public Broccoli() {
        super.name = "西蓝花";
    }
    
    @Override
    public void prepare() {
        System.out.println(super.name + "正在准备");
    }
}

生鲜厂

public class Factory {

    public static Fresh create(String type) {
        if (type.equals("broccoli")) {
            return new Broccoli();
        } else if (type.equals("cucumber")) {
            return new Cucumber();
        } else {
            return null;
        }
    }
}

河马生鲜

public class Boxhorse {
    public Boxhorse() {
        Fresh fresh = null;
        String orderType; // 订购生鲜产品
        do {
            fresh = Factory.create(getType());
            if (null != fresh) {
                fresh.prepare();
                fresh.washing();
                fresh.cut();
                fresh.box();
            } else {
                System.out.println("不卖了");
                break;
            }
        } while (true);
    }

    private String getType() {
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in));
        System.out.println("input type: ");
        try {
            return bufferedReader.readLine();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return "";
    }
}

Main

public class Main {
    public static void main(String[] args) {
        new Boxhorse();
    }
}

在这里解释一下每个类的作用

Fresh.java,生鲜食品

Cucumber.java, 黄瓜

Broccoli.java 西蓝花

Factory.java 生鲜加工厂。 这个类就是我们这个简单工厂的核心代码了,它存在的意义是统一加工生产生鲜产品。你只需要告诉他一个种类,便可以得到一个你想要的食品。当后面需要新增产品时,无需修改其他地方,只需要在工厂里新增几行代码即可。改动代价小

Boxhorse.java 河马生鲜。 或许你可能会认为这个有点多余,但是实际开发过程中,生鲜商店会有很多。就好比同城的盒马有很多。如果不统一加工,那每次加工都得新开一个工厂。无法集中管理

Main.java 入口

工厂方法模式

同城的生鲜可以在一个工厂加工,但是跨省市,就无法保证新鲜了。但是为了让工序变得固定,食品安全可靠。我们需要扩大规模

public abstract class Fresh {

    protected String name;

    public abstract void prepare();

    public void washing() {
        System.out.println(name + "正在清洗");
    }

    public void cut() {
        System.out.println(name + "正在分切");
    }

    public void box() {
        System.out.println(name + "正在打包");
    }

}
public class Panda extends Fresh {

    public Panda() {
        super.name = "鲜活大熊猫";
    }

    @Override
    public void prepare() {
        System.out.println(super.name + "正在清洗");
    }
}
public class Octopus extends Fresh {

    public Octopus() {
        super.name = "鲜活八爪鱼";
    }

    @Override
    public void prepare() {
        System.out.println(super.name + "正在清洗");
    }
}
public class Fungi extends Fresh {

    public Fungi() {
        super.name = "新鲜菌类";
    }

    @Override
    public void prepare() {
        System.out.println(super.name + "正在清洗");
    }
}
public class Crab extends Fresh {

    public Crab() {
        super.name = "鲜活梭子蟹";
    }

    @Override
    public void prepare() {
        System.out.println(super.name + "正在清洗");
    }
}

主要的四个类

public abstract class BaseFactory {
    public BaseFactory(String factoryName) {
        System.out.println(factoryName);
    }

    /**
     * 订购
     */
    public void shopping() {
        do {
            Fresh fresh = createFresh(getType());
            if (null != fresh) {
                fresh.prepare();
                fresh.cut();
                fresh.box();
            } else {
                System.out.println("没有,不卖了");
                break;
            }
        } while (true);
    }

    /**
     * 子类实现生鲜的制作,总部只负责调度,派发
     *
     * @param type 产品
     * @return 生鲜类
     */
    abstract Fresh createFresh(String type);

    private String getType() {
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in));
        System.out.println("input fresh type: ");
        try {
            return bufferedReader.readLine();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return "";
    }
}
public class CDFactory extends BaseFactory {

    public CDFactory() {
        super("成都工厂");
    }

    @Override
    Fresh createFresh(String type) {
        if ("fungi".equals(type)) {
            return new Fungi();
        } else if ("panda".equals(type)) {
            return new Panda();
        }
        return null;
    }
}
public class GZFactory extends BaseFactory {

    public GZFactory() {
        super("广州工厂");
    }

    @Override
    Fresh createFresh(String type) {
        if ("octopus".equals(type)) {
            return new Octopus();
        } else if ("crab".equals(type)) {
            return new Crab();
        }
        return null;
    }
}
public class Boxhorse {
    private BaseFactory factory;

    public Boxhorse() {
        // 需要初始化一个工厂
        this.factory = new CDFactory();
    }

    public void shopping() {
        if (null != this.factory) {
            this.factory.shopping();
        } else {
            System.out.println("没有选择工厂");
        }
    }
}

生鲜类就不解释了,这里主要变化的有四个类

BaseFactory.java 工厂总部。用户下单到总部,由总部下单到子工厂。

CDFactory.java 成都工厂。成都工厂负责成都市的生鲜制作与派发,主要产品有大熊猫、菌子

GZFactory.java 广州工厂。广州工厂负责广州市的生鲜制作与派发,主要产品有梭子蟹、八爪鱼

Boxhorse.java 生鲜超市。一个生鲜超市配有一个工厂

public class Main {
    public static void main(String[] args) {
        new Boxhorse().shopping();
        new Boxhorse2().shopping();
    }
}

console

成都工厂
input fresh type:
panda
鲜活大熊猫正在清洗
鲜活大熊猫正在分切
鲜活大熊猫正在打包
input fresh type:
crab
没有,不卖了
广州工厂
input fresh type:
crab
鲜活梭子蟹正在清洗
鲜活梭子蟹正在分切
鲜活梭子蟹正在打包
input fresh type:
exit
没有,不卖了

Process finished with exit code 0

下面是拓展,加入了单例模式

所有工厂必须取得证书才可以生产出产品,证书有限,只有两个。但是却有三个厂

由基类(工厂总部)进行证书的派发与校验。实际上这里才能感受到工厂方法模式最大的省事之处。

enum Authorization {
    INSTANCE;
    List<String> authorized = new ArrayList<>();
    List<String> certificate = new ArrayList<>(Arrays.asList("A1", "A2"));


    /**
     * 取得证书
     * @return 证书
     */
    String getAuth() {
        String auth = null;
        if (certificate.size() > 0) {
            auth = certificate.get(0);
            authorized.add(auth);
            certificate.remove(0);
        }
        return auth;
    }

    /**
     * 检查证书
     * @param auth 证书
     * @return 是否存在
     */
    boolean check(String auth) {
        return authorized.contains(auth);
    }
}

public abstract class BaseFactory{
    private boolean admit = false;
    private String factoryName;

    public BaseFactory(String auth, String factoryName) {
        this.factoryName = factoryName;
        this.admit = Authorization.INSTANCE.check(auth);
    }

    public void shopping() {
        if (admit) {
            System.out.println(factoryName + "取得认证");
            do {
                Fresh fresh = createFresh(getType());
                if (null != fresh) {
                    fresh.prepare();
                    fresh.cut();
                    fresh.box();
                } else {
                    System.out.println("没有,不卖了");
                    break;
                }
            } while (true);
        } else {
            System.out.println(factoryName + "没有取得认证,结束");
        }
    }

    /**
     * 子类实现生鲜
     * @param type 产品
     * @return 生鲜类
     */
    abstract Fresh createFresh(String type);

    /**
     * 各个工厂都需要获得证书才可以生产生鲜
     * @return String
     */
    protected static String authorization() {
        return Authorization.INSTANCE.getAuth();
    }

    private String getType() {
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in));
        System.out.println("input fresh type: ");
        try {
            return bufferedReader.readLine();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return "";
    }
}
public class CDFactory extends BaseFactory {

    public CDFactory() {
        // 调用父类的构造器,并从父类取得证书。传递给父类
        super(authorization(), "成都工厂");
    }

    @Override
    Fresh createFresh(String type) {
        if ("fungi".equals(type)) {
            return new Fungi();
        } else if ("panda".equals(type)) {
            return new Panda();
        }
        return null;
    }
}
public class GZFactory extends BaseFactory {

    public GZFactory() {
        // 调用父类的构造器,并从父类取得证书。传递给父类
        super(authorization(), "广州工厂");
    }

    @Override
    Fresh createFresh(String type) {
        if ("octopus".equals(type)) {
            return new Octopus();
        } else if ("crab".equals(type)) {
            return new Crab();
        }
        return null;
    }
}
public class ZLFactory extends BaseFactory {

    public ZLFactory() {
        super(authorization(), "脏乱工厂");
    }

    @Override
    Fresh createFresh(String type) {
        if ("octopus".equals(type)) {
            return new Octopus();
        } else if ("crab".equals(type)) {
            return new Crab();
        }
        return null;
    }
}




public class Main {
    public static void main(String[] args) {
        new Boxhorse().shopping();
        new Boxhorse2().shopping();
        new Boxhorse3().shopping();
    }
}

console

成都工厂取得认证
input fresh type:
crab
没有,不卖了
广州工厂取得认证
input fresh type:
crab
鲜活梭子蟹正在清洗
鲜活梭子蟹正在分切
鲜活梭子蟹正在打包
input fresh type:
ex
没有,不卖了
脏乱工厂没有取得认证,结束

Process finished with exit code 0

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

推荐阅读更多精彩内容