1. 定义(我都看了,觉得还不错,留下来帮助理解,可以挑着看)
工厂方法(FactoryMethod)模式的定义:定义一个创建产品对象的工厂接口,将产品对象的实际创建工作推迟到具体子工厂类当中。这满足创建型模式中所要求的“创建与使用相分离”的特点。
工厂模式它提供了一种创建对象的最佳方式。在工厂模式中,我们在创建对象时不会对客户端暴露创建逻辑,并且是通过使用一个共同的接口来指向新创建的对象。
工厂模式是我们最常用的实例化对象模式了,是用工厂方法代替new操作的一种模式。因为工厂模式就相当于创建实例对象的new,我们经常要根据类Class生成实例对象,如A a=new A() 工厂模式也是用来创建实例对象的,所以以后new时就要多个心眼,是否可以考虑使用工厂模式,虽然这样做,可能多做一些工作,但会给你系统带来更大的可扩展性和尽量少的修改量。
工厂模式定义
我们以类Sample为例, 如果我们要创建Sample的实例对象:
Sample sample=new Sample();
可是,实际情况是,通常我们都要在创建sample实例时做点初始化的工作,比如赋值 查询数据库等。
首先,我们想到的是,可以使用Sample的构造函数,这样生成实例就写成:
Sample sample=new Sample(参数);
但是,如果创建sample实例时所做的初始化工作不是像赋值这样简单的事,可能是很长一段代码,如果也写入构造函数中,那你的代码很难看了(就需要Refactor重构)。
为什么说代码很难看,初学者可能没有这种感觉,我们分析如下,初始化工作如果是很长一段代码,说明要做的工作很多,将很多工作装入一个方法中,相当于将很多鸡蛋放在一个篮子里,是很危险的,这也是有悖于Java面向对象的原则,面向对象的封装(Encapsulation)和分派(Delegation)告诉我们,尽量将长的代码分派“切割”成每段,将每段再“封装”起来(减少段和段之间耦合联系性),这样,就会将风险分散,以后如果需要修改,只要更改每段,不会再发生牵一动百的事情。
在本例中,首先,我们需要将创建实例的工作与使用实例的工作分开,
也就是说,让创建实例所需要的大量初始化工作从Sample的构造函数中分离出去。
这时我们就需要Factory工厂模式来生成对象了,不能再用上面简单new
Sample(参数)。还有,如果Sample有个继承如MySample,
按照面向接口编程,我们需要将Sample抽象成一个接口.Sample是接口,有两个子类MySample 和HisSample
.我们要实例化他们时,如下: ISample mysample=new MySample(); ISample hissample=new
HisSample();
随着项目的深入,Sample可能还会"生出很多儿子出来",
那么我们要对这些儿子一个个实例化,更糟糕的是,可能还要对以前的代码进行修改:加入后来生出儿子的实例.这在传统程序中是无法避免的.
但如果你一开始就有意识使用了工厂模式,这些麻烦就没有了.
2.模式的结构与实现
工厂方法模式由抽象工厂、具体工厂、抽象产品和具体产品等4个要素构成。本节来分析其基本结构和实现方法。
1.模式的结构(4个主要角色)
- 抽象工厂(Abstract Factory):提供了创建产品的接口,调用者通过它访问具体工厂的工厂方法 newProduct() 来创建产品。
- 具体工厂(ConcreteFactory):主要是实现抽象工厂中的抽象方法,完成具体产品的创建。
- 抽象产品(Product):定义了产品的规范,描述了产品的主要特性和功能。
- 具体产品(ConcreteProduct):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间一一对应。
让其子类实现工厂接口,返回的也是一个抽象的产品。
创建过程在其子类执行。
2. 模型的实现
//抽象产品:提供了产品的接口
public interface Laptop {
public void getLaptop();
}
//具体产品1:实现抽象产品中的抽象方法
public class AcerLaptop implements Laptop {
@Override
public void getLaptop() {
System.out.println("Acer笔记本");
}
}
//具体产品2:实现抽象产品中的抽象方法
public class HpLaptop implements Laptop {
@Override
public void getLaptop() {
System.out.println("惠普笔记本");
}
}
//抽象工厂:提供了厂品的生成方法
public interface AbstractFactory {
public Laptop newProduct();
}
//具体工厂1:实现了厂品的生成方法
public class AcerFactory implements AbstractFactory{
@Override
public Laptop newProduct() {
System.out.println("Acer工厂生产Acer笔记本");
return new AcerLaptop();
}
}
//具体工厂2:实现了厂品的生成方法
public class HpFactory implements AbstractFactory {
@Override
public Laptop newProduct() {
System.out.println("hp工厂生产惠普笔记本");
return new HpLaptop();
}
}
测试
public class Test {
public static void main(String[] args) {
AbstractFactory hpfactory模式的应用场景 = new HpFactory();
AbstractFactory acerfactory = new AcerFactory();
Laptop hplaptop = hpfactory.newProduct();
Laptop acerlaptop = acerfactory.newProduct();
hplaptop.getLaptop();
acerlaptop.getLaptop();
}
}
*模式的应用场景
工厂方法模式通常适用于以下场景。
-
客户只知道创建产品的工厂名,而不知道具体的产品名。如 TCL 电视工厂、海信电视工厂等。
-
创建对象的任务由多个具体子工厂中的某一个完成,而抽象工厂只提供创建产品的接口。
-
客户不关心创建产品的细节,只关心产品的品牌。
模式的扩展
当需要生成的产品不多且不会增加,一个具体工厂类就可以完成任务时,可删除抽象工厂类
。这时工厂方法模式将退化到简单工厂模式