设计模式 _第二招式_工厂方法模式

本文深入讲解了工厂方法模式,一种常见的设计模式,用于定义创建对象的接口,让子类决定实例化哪个类,实现了对象创建的封装和扩展。文章详细介绍了模式的结构,包括抽象产品类、抽象工厂和具体工厂等组件,并提供了代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、定义
工厂方法模式使用的频率非常高,在我们日常的开发中总能看到它的身影。其定义为:定义一个用于创建对象的接口,让子类决定实例化哪一个类。工程方法使用一个类的实例化延迟到其子类。
二、代码演示
在工厂方法模式中, 抽象产品类Product负责定义产品的共性,实现对事物最抽象的定义;Creater为抽象创建类,已经是抽象工程,具体如何创建产品类是由具体的实现工程ConcreteCreator完成的。工程方法的变种比较多,下面是一个比较通用的源代码。
工厂方法模式通用的类图,如下:
这里写图片描述

  • 抽象产品类
public abstract class Product {
   //产品类的公共方法
    public  void method1(){
        //业务逻辑
    }
    //抽象方法
    public abstract void method2();
}
  • 具体产品类

具体的产品可以有多个,都继承于抽象产品类,其源代码如下:

public class ConcreteProduct1 extends  Product {
    public void  method2(){
        //业务逻辑处理
    }
}
public class ConcreteProduct2 extends Product {
    public void method2(){
        //业务逻辑处理
    }
}
  • 抽象工厂

抽象工程负责定义产品对象的产生,源码如下:

public abstract class Creator {
    public abstract <T extends Product > T createProduct(Class<T> c);
}
  • 具体工厂类

具体如何产生一个产品的对象,是由具体工厂实现的,代码如下:

public class ConcreteCreator extends  Creator {
    public  <T extends  Product> T createProduct(Class<T> c){
        Product product = null ;
        try {
            product =(Product)Class.forName(c.getName()).newInstance();
        }catch (Exception e){

        }
        return  (T)product;
    }
}
  • 场景类

场景负责调用方法,代码如下:

public class Client {
    public  static  void  main(String args[]){
        Creator  creator = new ConcreteCreator();
        Product  product = creator.createProduct(ConcreteProduct1.class);
        /**
         * 继续业务处理
         */
    }
}

三、优点

  1. 良好的封装性,代码结构清晰。一个对象创建有条件约束的,如一个调用者需要是一个具体的产品对象,只要知道这个产品的类名(或约束字符串)就可以了。
    不需要知道过程,降低模块间的耦合。
  2. 工厂方法模式的扩展性非常优秀。在增加产品的情况下,只要适当修改具体的工程类或扩展一个工程类,就可以“拥抱变化”。
  3. 屏蔽产品类。这一特点非常重要,产品类的实现如何变化,调用者都不需要关系,只需要关心产品的接口,只要接口保持不变,系统中的上层模块就不要发生变化。在数据库开发中,应该能体会到工厂方法模式的好处,我们要将mysql
    数据库, 切换到oracle 数据库,只需要修改一下驱动名称,这就是工厂方法模式的案例。
  4. 工厂方法模式是典型的解耦框架。高层模块值需要知道产品的抽象类,其他的实现类都不用关心,符合迪米特法则,我不需要的就不要去交流;也符合依赖倒置原则,值依赖产品类的抽象;也符合里氏替换原则,使用产品子类替换为父类。

四、缺点
需要谨慎考虑是否需要增加一个工厂类管理,增加代码的复杂度。

五、应用场景

  • 工厂方法模式是new一个对象的替代品,所以在所有需要生成对象的地方都可以使用,但是需要谨慎考虑是否需要增加一个工厂类管理,增加代码的复杂度。
  • 需要灵活的、可扩展的框架时,可以考虑工厂方法模版。万物皆对象,那万物就是产品类,例如需要设计一个连接邮件服务器的框架,有三种网络协议可选择:POP3、IMAP、HTTP,我们就把这三种连接方法作为产品类,定义一个借口如IConnectMail,然后定义邮件操作的方法,用不同的方法实现三个具体的产品类,在定义一个工厂方法,按照不同的传入条件,选择不同的连接方式。如此设计可做到完美扩展,如某些服务器提供webservice接口,很好,我们只要增加一个产品类就可以了。
  • 工厂方法模版可以用在异构项目中,例如通过webservice与一个非java的项目交互,虽然webservice号称是可以做到异构系统的同构化,但是实际开发中还是会碰到很多问题,如类型问题、WSDL文件的支持问题,等等。从WSDL中产生的对象都认为是一个产品,然后有一个具体的工程进行管理,减少与外围的耦合。
  • 可以使用测试驱动开发的框架下。例如,测试一个类A,就需要把与类A有关联关系的类B也同步生产出来,我们可以用工厂方法模版把类B虚拟出来,避免类A与类B的耦合。目前由于JMock和EasyMock的诞生,该使用场景已经弱化,读者可以在遇到此情况时候直接考虑使用JMock和EasyMock。

六、注意事项

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值