Java设计模式圣经连载(02)-工厂方法(Factory Method)模式
工厂方法模式是类的创建模式,又叫虚拟构造子(Virtual Constructor)模式或者多态性工厂(Polymorphic Factory)模式。
工厂方法模式的用意是定义一个创建产品对象的工厂接口,将实际工作推迟到子类中。
在工厂方法模式中,核心的工厂类不再负责所有产品的创建,而是将具体创建的工作交给子类去做。这个核心工厂则变为抽象工厂角色,仅负责给出具工厂子类必须实现的接口,而不接触哪一产品创建的细节。
这种抽象的结果,使这种工厂方法模式可以用来允许系统不修改具体工厂角色的情况下引进新产品,这一特点无疑使得工厂模式具有超过简单工厂模式的优越性。
从上图可以看出,工厂方法模式的系统涉及到了以下角色:
抽象工厂角色:与应用程序无关,任何在模式中创建对象的工厂必须实现这个接口。
具体工厂角色:实现了抽象工厂接口的具体Java类,含有与引用密切相关的逻辑,并且受到应用程序的调用以创建产品对象。
抽象产品角色:工厂方法所创建产品对象的超类型,也就是产品对象的共同父类或共同拥有的接口。
具体产品角色:这个角色实现了抽象产品角色所声名的接口。工厂方法所创建的每个具体产品对象都是某个具体产品角色的实例。
1.2.3.3 实现源码
1.2.3.3.3 具体产品葡萄:Grape.java
package com.lavasoft.patterns.factorymethod.ybms;
1.2.3.3.4 具体产品草莓:Strawberry.java
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:Strawberry.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:草莓
*/
public class Strawberry implements Fruit {
/**
* 生长
*/
public void grow() {
System.out.println("Strawberry is growing...");
}
呵呵,学会了简单工厂模式再来学工厂方法,是不是很容易啊,看懂农场一个例子就够了,多实践多思考比什么都好!
工厂方法模式的用意是定义一个创建产品对象的工厂接口,将实际工作推迟到子类中。
1.2.1 工厂方法模式的引进
工厂方法模式是简单工厂模式的进一步抽象和推广。由于使用了多态性,工厂方法模式保持了简单工厂模式的优点,而且克服了它的缺点。下面是工厂方法模式的缩略类图:
工厂方法模式是简单工厂模式的进一步抽象和推广。由于使用了多态性,工厂方法模式保持了简单工厂模式的优点,而且克服了它的缺点。下面是工厂方法模式的缩略类图:
在工厂方法模式中,核心的工厂类不再负责所有产品的创建,而是将具体创建的工作交给子类去做。这个核心工厂则变为抽象工厂角色,仅负责给出具工厂子类必须实现的接口,而不接触哪一产品创建的细节。
这种抽象的结果,使这种工厂方法模式可以用来允许系统不修改具体工厂角色的情况下引进新产品,这一特点无疑使得工厂模式具有超过简单工厂模式的优越性。
1.2.2 工厂方法模式的结构
在工厂方法模式中,一般都有一个平行的等级结构,也就是说工厂和产品是对应的的。抽象工厂对应抽象产品,具体工厂对应具体产品。简单的示意图如下:
在工厂方法模式中,一般都有一个平行的等级结构,也就是说工厂和产品是对应的的。抽象工厂对应抽象产品,具体工厂对应具体产品。简单的示意图如下:
从上图可以看出,工厂方法模式的系统涉及到了以下角色:
抽象工厂角色:与应用程序无关,任何在模式中创建对象的工厂必须实现这个接口。
具体工厂角色:实现了抽象工厂接口的具体Java类,含有与引用密切相关的逻辑,并且受到应用程序的调用以创建产品对象。
抽象产品角色:工厂方法所创建产品对象的超类型,也就是产品对象的共同父类或共同拥有的接口。
具体产品角色:这个角色实现了抽象产品角色所声名的接口。工厂方法所创建的每个具体产品对象都是某个具体产品角色的实例。
1.2.3 工厂方法模式在农场系统中的实现
1.2.3.1 背景
在简单工厂模式中,有个全能的园丁,控制所有作物的种植、生长和收获。现在农场规模变大了,管理更加专业化了。过去全能的园丁没有了,每一种作物都有专门的园丁管理,形成了规模化和专业化生产。
在简单工厂模式中,有个全能的园丁,控制所有作物的种植、生长和收获。现在农场规模变大了,管理更加专业化了。过去全能的园丁没有了,每一种作物都有专门的园丁管理,形成了规模化和专业化生产。
1.2.3.2 系统设计机构图
1.2.3.3 实现源码
1.2.3.3.1 水果产品接口Fruit.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:Fruit.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果接口
*/
public interface Fruit {
/**
* 种植
*/
void plant();
* Created by IntelliJ IDEA.
* FileName:Fruit.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果接口
*/
public interface Fruit {
/**
* 种植
*/
void plant();
/**
* 生长
*/
void grow();
* 生长
*/
void grow();
/**
* 收获
*/
void harvest();
}
* 收获
*/
void harvest();
}
1.2.3.3.2 具体产品苹果Apple.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:Apple.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:苹果
*/
public class Apple implements Fruit {
private int treeAge;
* Created by IntelliJ IDEA.
* FileName:Apple.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:苹果
*/
public class Apple implements Fruit {
private int treeAge;
/**
* 种植
*/
public void plant() {
System.out.println("Apple has been planted.");
}
* 种植
*/
public void plant() {
System.out.println("Apple has been planted.");
}
/**
* 生长
*/
public void grow() {
System.out.println("Apple is growing...");
}
* 生长
*/
public void grow() {
System.out.println("Apple is growing...");
}
/**
* 收获
*/
public void harvest() {
System.out.println("Apple has been harvested.");
}
* 收获
*/
public void harvest() {
System.out.println("Apple has been harvested.");
}
/**
* @return 返回树龄
*/
public int getTreeAge() {
return treeAge;
}
* @return 返回树龄
*/
public int getTreeAge() {
return treeAge;
}
/**
* 设置树龄
*/
public void setTreeAge(int treeAge) {
this.treeAge = treeAge;
}
* 设置树龄
*/
public void setTreeAge(int treeAge) {
this.treeAge = treeAge;
}
}
1.2.3.3.3 具体产品葡萄:Grape.java
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:Grape.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:葡萄
*/
public class Grape implements Fruit {
private boolean seedless; //是否有籽
* Created by IntelliJ IDEA.
* FileName:Grape.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:葡萄
*/
public class Grape implements Fruit {
private boolean seedless; //是否有籽
/**
* 种植
*/
public void plant() {
System.out.println("Grape has been planted.");
}
* 种植
*/
public void plant() {
System.out.println("Grape has been planted.");
}
/**
* 生长
*/
public void grow() {
System.out.println("Grape is growing...");
}
* 生长
*/
public void grow() {
System.out.println("Grape is growing...");
}
/**
* 收获
*/
public void harvest() {
System.out.println("Grape has been harvested.");
}
* 收获
*/
public void harvest() {
System.out.println("Grape has been harvested.");
}
/**
* @return 是否有籽
*/
public boolean getSeedless() {
return seedless;
}
* @return 是否有籽
*/
public boolean getSeedless() {
return seedless;
}
/**
* 有无籽的赋值方法
*/
public void setSeedless(boolean seedless) {
this.seedless = seedless;
}
* 有无籽的赋值方法
*/
public void setSeedless(boolean seedless) {
this.seedless = seedless;
}
/**
* 辅助方法
*/
public static void log(String msg) {
System.out.println(msg);
}
}
* 辅助方法
*/
public static void log(String msg) {
System.out.println(msg);
}
}
1.2.3.3.4 具体产品草莓:Strawberry.java
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:Strawberry.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂的产品:草莓
*/
public class Strawberry implements Fruit {
/**
* 生长
*/
public void grow() {
System.out.println("Strawberry is growing...");
}
/**
* 收获
*/
public void harvest() {
System.out.println("Strawberry has been harvested.");
}
* 收获
*/
public void harvest() {
System.out.println("Strawberry has been harvested.");
}
/**
* 种植
*/
public void plant() {
System.out.println("Strawberry has been planted.");
}
* 种植
*/
public void plant() {
System.out.println("Strawberry has been planted.");
}
/**
* 辅助方法
*/
public static void log(String msg) {
System.out.println(msg);
}
}
* 辅助方法
*/
public static void log(String msg) {
System.out.println(msg);
}
}
1.2.3.3.5 水果工厂接口:FruitGardener.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:FruitGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂接口
*/
public interface FruitGardener {
/**
* 工厂方法
*
* @return 水果
*/
public Fruit factory();
}
* Created by IntelliJ IDEA.
* FileName:FruitGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 水果工厂接口
*/
public interface FruitGardener {
/**
* 工厂方法
*
* @return 水果
*/
public Fruit factory();
}
1.2.3.3.6 苹果工厂:AppleGardener.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:AppleGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 苹果工厂方法
*/
public class AppleGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 苹果
*/
public Fruit factory() {
Fruit f = new Apple();
System.out.println("水果工厂(AppletGardener)成功创建一个水果:苹果!");
return f;
}
}
* Created by IntelliJ IDEA.
* FileName:AppleGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 苹果工厂方法
*/
public class AppleGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 苹果
*/
public Fruit factory() {
Fruit f = new Apple();
System.out.println("水果工厂(AppletGardener)成功创建一个水果:苹果!");
return f;
}
}
1.2.3.3.7 葡萄工厂:GrapeGardener.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:GrapeGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 添加说明
*/
public class GrapeGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 葡萄
*/
public Fruit factory() {
Fruit f = new Grape();
System.out.println("水果工厂(GrapeGardener)成功创建一个水果:葡萄!");
return f;
}
}
* Created by IntelliJ IDEA.
* FileName:GrapeGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 添加说明
*/
public class GrapeGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 葡萄
*/
public Fruit factory() {
Fruit f = new Grape();
System.out.println("水果工厂(GrapeGardener)成功创建一个水果:葡萄!");
return f;
}
}
1.2.3.3.8 草莓工厂:StrawberryGardener.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:StrawberryGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 添加说明
*/
public class StrawberryGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 草莓
*/
public Fruit factory() {
Fruit f = new Strawberry();
System.out.println("水果工厂(StrawberryGardener)成功创建一个水果:草莓!");
return f;
}
}
* Created by IntelliJ IDEA.
* FileName:StrawberryGardener.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 添加说明
*/
public class StrawberryGardener implements FruitGardener {
/**
* 工厂方法
*
* @return 草莓
*/
public Fruit factory() {
Fruit f = new Strawberry();
System.out.println("水果工厂(StrawberryGardener)成功创建一个水果:草莓!");
return f;
}
}
1.2.3.3.9 测试类(客户端):TestApp.java
package com.lavasoft.patterns.factorymethod.ybms;
package com.lavasoft.patterns.factorymethod.ybms;
/**
* Created by IntelliJ IDEA.
* FileName:TestApp.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 测试类(客户端)
*/
public class TestApp {
private FruitGardener f1, f2, f3;
private Fruit p1, p2, p3;
* Created by IntelliJ IDEA.
* FileName:TestApp.java
* 工厂模式--工厂方法模式--一般性模式(农场应用)
* ReadMe: 测试类(客户端)
*/
public class TestApp {
private FruitGardener f1, f2, f3;
private Fruit p1, p2, p3;
private void test() {
//实力化水果工厂
f1 = new AppleGardener();
f2 = new GrapeGardener();
f3 = new StrawberryGardener();
//从水果工厂生产水果
p1 = f1.factory();
p2 = f2.factory();
p3 = f3.factory();
}
//实力化水果工厂
f1 = new AppleGardener();
f2 = new GrapeGardener();
f3 = new StrawberryGardener();
//从水果工厂生产水果
p1 = f1.factory();
p2 = f2.factory();
p3 = f3.factory();
}
public static void main(String args[]) {
TestApp test = new TestApp();
test.test();
}
}
TestApp test = new TestApp();
test.test();
}
}
1.2.3.3.10 测试运行结果
水果工厂(AppletGardener)成功创建一个水果:苹果!
水果工厂(GrapeGardener)成功创建一个水果:葡萄!
水果工厂(StrawberryGardener)成功创建一个水果:草莓!
水果工厂(AppletGardener)成功创建一个水果:苹果!
水果工厂(GrapeGardener)成功创建一个水果:葡萄!
水果工厂(StrawberryGardener)成功创建一个水果:草莓!
Process finished with exit code 0
1.2.3.4 关于工厂方法模式的实现
在下面我简单说说一些实现方式,不做详细分析了。
1.使用java接口或者java抽象类
2.使用多个工厂方法
3.产品循环使用
4.多态性的丧失和模式的退化,主要体现在工厂方法创建对象、工厂方法返回的类型、工厂等级结构三个方面。
在下面我简单说说一些实现方式,不做详细分析了。
1.使用java接口或者java抽象类
2.使用多个工厂方法
3.产品循环使用
4.多态性的丧失和模式的退化,主要体现在工厂方法创建对象、工厂方法返回的类型、工厂等级结构三个方面。
1.2.3.5 女娲举绳造人
女娲举绳造人是工厂方法模式的一个实例。下面仅仅给出设计图如下,具体编码请您自己实现。
女娲举绳造人是工厂方法模式的一个实例。下面仅仅给出设计图如下,具体编码请您自己实现。
呵呵,学会了简单工厂模式再来学工厂方法,是不是很容易啊,看懂农场一个例子就够了,多实践多思考比什么都好!