《23种设计模式(Java版)》| 工厂模式(内附源码案例)。

简介: 工厂模式是我们最常用的实例化对象模式了,是用工厂方法代替new操作的一种模式。著名的Jive论坛 ,就大量使用了工厂模式,工厂模式在Java程序系统可以说是随处可见。因为工厂模式就相当于创建实例对象的new,我们经常要根据类Class生成实例对象,如A a=new A() 工厂模式也是用来创建实例对象的,所以以后new时就要多个心眼,是否可以考虑使用工厂模式,虽然这样做,可能多做一些工作,但会给你系统带来更大的可扩展性和尽量少的修改量

1.png

💎 一、概述


工厂模式是我们最常用的实例化对象模式了,是用工厂方法代替new操作的一种模式。著名的Jive论坛 ,就大量使用了工厂模式,工厂模式在Java程序系统可以说是随处可见。因为工厂模式就相当于创建实例对象的new,我们经常要根据类Class生成实例对象,如A a=new A() 工厂模式也是用来创建实例对象的,所以以后new时就要多个心眼,是否可以考虑使用工厂模式,虽然这样做,可能多做一些工作,但会给你系统带来更大的可扩展性和尽量少的修改量


📕 二、工厂模式的三种方式


🏭 简单工厂

简单工厂模式是属于创建型模式,是工厂模式的一种。简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。简单工厂模式是工厂模式家族中最简单实用的模式

简单工厂模式:定义了一个创建对象的类,由这个类来封装实例化对象的行为(代码)

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


案例代码:


//定义一个公共接口
public interface Car {
  void gotoWork();
}
public class Bus implements Car{
  @Override
  public void gotoWork() {
    // TODO Auto-generated method stub
    System.out.println("坐公交上班");
  }
}
public class Bike implements Car{
  @Override
  public void gotoWork() {
    // TODO Auto-generated method stub
    System.out.println("骑自行车上班");
  }
}
//简单工厂类
public class SimpleFactory {
  public enum CarType{
    Bike,Bus;
  }
  //通过不同的方式构建不同的实例
  public static Car getCar(CarType car) {
    Car simpleCar=null;
    switch(car) {
    case Bike:
      simpleCar=new Bike();
      break;
    case Bus:
      simpleCar= new Bus();
      break;
      default:
       simpleCar=new Bike();
    }
    return simpleCar;
  }
}

🔨 工厂方法

通过一个需求案例:


看一个披萨的项目:要便于披萨种类的扩展,要便于维护

1.披萨的种类很多(比如 GreekPizz、CheesePizz 等

2.披萨的制作有 prepare,bake, cut, box

3.完成披萨店订购功能

客户在点披萨时,可以点不同口味的披萨,比如 北京的奶酪pizza、

北京的胡椒pizza 或者是伦敦的奶酪pizza、伦敦的胡椒pizza。


工厂方法模式设计方案:将披萨项目的实例化功能抽象成抽象方法,在不同的口味点餐子类中具体实现。

工厂方法模式:定义了一个创建对象的抽象方法,由子类决定要实例化的类。工厂方法模式将对象的实例化推迟到子类

1.png

案例代码:


public abstract class Pizza {
  protected String name;
  //准备材料,不同的披萨材料不一样,做成抽象方法
  public abstract void prepare();
  public void bake() {
    System.out.println(name + " baking;");
  }
  public void cut() {
    System.out.println(name + " cutting;");
  }
  public void box() {
    System.out.println(name + " boxing;");
  }
  public void setName(String name) {
    this.name = name;
  }
}
public class BJCheesePizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("北京的奶酪披萨");
    System.out.println("北京的奶酪披萨准备原材料");
  }
}
public class BJGreekPizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("北京的希腊披萨");
    System.out.println("北京的希腊披萨准备原材料");
  }
}
public class LDCheesePizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("伦敦奶酪披萨");
    System.out.println("伦敦奶酪披萨准备原材料");
  }
}
public class LDGreekPizz extends Pizza
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("伦敦希腊披萨");
    System.out.println("伦敦希腊披萨准配原材料");
  }
}
public abstract class OrderPizza {
   //定义一个抽象方法,让各个工厂子类自己实现
   abstract Pizza creatPizza(String orderType);
   //构造器
   public OrderPizza() {
    Pizza pizza = null;
    // 订购披萨类型
    String orderType;
    do {
      orderType = getType();
      pizza=creatPizza(orderType);
      pizza.prepare();
      pizza.bake();
      pizza.cut();
      pizza.box();
    } while (true);
  }
    // 动态的获取用户希望订购的披萨
  private String getType() {
    try {
      BufferedReader strin = new BufferedReader(new InputStreamReader(System.in));
      System.out.println("input pizza type:");
      String str = strin.readLine();
      return str;
    } catch (IOException e) {
      e.printStackTrace();
      return "";
    }
  }
}
public class BJOrderPizza extends OrderPizza{
  @Override
  Pizza creatPizza(String orderType) {
    Pizza pizza=null;
    if(orderType.equals("cheese")) {
      pizza=new BJCheesePizz();
    }else if(orderType.equals("greek")) {
      pizza=new BJGreekPizz();
    }
    return pizza;
  }
}
public class LDOrderPizza extends OrderPizza{
  @Override
  Pizza creatPizza(String orderType) {
    Pizza pizza=null;
    if(orderType.equals("cheese")) {
      pizza=new LDCheesePizz();
    }else if(orderType.equals("greek")) {
      pizza=new LDGreekPizz();
    }
    return pizza;
  }
}
public class PizzaStore {
  public static void main(String[] args) {
    // TODO Auto-generated method stub
    //创建北京口味的披萨
//    new BJOrderPizza();
    //创建伦敦口味的披萨
    new LDOrderPizza();
  }
}


🪓 抽象工厂


抽象工厂模式:定义了一个interface用于创建相关或有依赖关系的对象簇,而无需指明具体的类

抽象工厂模式可以将简单工厂模式和工厂方法模式进行整合。

从设计层面看,抽象工厂模式就是对简单工厂模式的改进(或者称为进一步的抽象)。

将工厂抽象成两层,AbsFactory(抽象工厂) 和具体实现的工厂子类。程序员可以根据创建对象类型使用对应的工厂子类。这样将单个的简单工厂类变成了工厂簇,更利于代码的维护和扩展。

1.png

⏰需求案例同工厂方法案例,抽象工厂对其进行进一步的优化。


案例代码:


public abstract class Pizza {
  protected String name;
  //准备材料,不同的披萨材料不一样,做成抽象方法
  public abstract void prepare();
  public void bake() {
    System.out.println(name + " baking;");
  }
  public void cut() {
    System.out.println(name + " cutting;");
  }
  public void box() {
    System.out.println(name + " boxing;");
  }
  public void setName(String name) {
    this.name = name;
  }
}
public class BJCheesePizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("北京的奶酪披萨");
    System.out.println("北京的奶酪披萨准备原材料");
  }
}
public class BJGreekPizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("北京的希腊披萨");
    System.out.println("北京的希腊披萨准备原材料");
  }
}
public class LDCheesePizz extends Pizza{
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("伦敦奶酪披萨");
    System.out.println("伦敦奶酪披萨准备原材料");
  }
}
public class LDGreekPizz extends Pizza
  @Override
  public void prepare() {
    // TODO Auto-generated method stub
    setName("伦敦希腊披萨");
    System.out.println("伦敦希腊披萨准配原材料");
  }
}
//抽象工厂模式的抽象层
public interface AbsFactory {
  //下面工厂子类自己建造
  Pizza creatPizza(String orderType);
}
public class BJFactory implements AbsFactory{
  public Pizza creatPizza(String orderType) {
    System.out.println("使用抽象工厂模式");
    // TODO Auto-generated method stub
    Pizza pizza=null;
    if(orderType.equals("cheese")) {
      pizza=new BJCheesePizz();
    }else if(orderType.equals("greek")) {
      pizza=new BJGreekPizz();
    }
    return pizza;
  }
}
public class LDFactory implements AbsFactory{
  public Pizza creatPizza(String orderType) {
    System.out.println("使用抽象工厂模式");
    // TODO Auto-generated method stub
    Pizza pizza=null;
    if(orderType.equals("cheese")) {
      pizza=new LDCheesePizz();
    }else if(orderType.equals("greek")) {
      pizza=new LDGreekPizz();
    }
    return pizza;
  }
public class OrderPizza {
  AbsFactory absFactory;
  public OrderPizza(AbsFactory absFactory) {
    setAbsFactory(absFactory);
  }
  private void setAbsFactory(AbsFactory absFactory) {
    Pizza pizza=null;
    String orderType="";
    this.absFactory = absFactory;
    do {
      orderType=getType();
      //absFactory 可能为北京 或者伦敦
      pizza=absFactory.creatPizza(orderType);
      if(pizza!=null) {
        pizza.prepare();
        pizza.bake();
        pizza.cut();
        pizza.box();
      }else {
        System.out.println("订购失败");
        break;
      }
    }while(true);
  }
  private String getType() {
    try {
      BufferedReader strin = new BufferedReader(new InputStreamReader(System.in));
      System.out.println("input pizza type:");
      String str = strin.readLine();
      return str;
    } catch (IOException e) {
      e.printStackTrace();
      return "";
    }
  }
}
public class PizzaStore {
  public static void main(String[] args) {
    // TODO Auto-generated method stub
      new OrderPizza(new BJFactory());
  }
}


📄 三、在JDK源码中的应用分析


JDK 中的Calendar类中,就使用了简单工厂模式


public abstract class Calendar implements Serializable, Cloneable, Comparable<Calendar> {
    public static Calendar getInstance(TimeZone zone,
                                       Locale aLocale){
        return createCalendar(zone, aLocale);
    }
    private static Calendar createCalendar(TimeZone zone,
                                           Locale aLocale){
        CalendarProvider provider =
            LocaleProviderAdapter.getAdapter(CalendarProvider.class, aLocale)
                                 .getCalendarProvider();
        if (provider != null) {
            try {
                return provider.getInstance(zone, aLocale);
            } catch (IllegalArgumentException iae) {
                // fall back to the default instantiation
            }
        }
        Calendar cal = null;
        if (aLocale.hasExtensions()) {
            String caltype = aLocale.getUnicodeLocaleType("ca");
            if (caltype != null) {
                switch (caltype) {
                case "buddhist":
                cal = new BuddhistCalendar(zone, aLocale);
                    break;
                case "japanese":
                    cal = new JapaneseImperialCalendar(zone, aLocale);
                    break;
                case "gregory":
                    cal = new GregorianCalendar(zone, aLocale);
                    break;
                }
            }
        }
        if (cal == null) {
            // If no known calendar type is explicitly specified,
            // perform the traditional way to create a Calendar:
            // create a BuddhistCalendar for th_TH locale,
            // a JapaneseImperialCalendar for ja_JP_JP locale, or
            // a GregorianCalendar for any other locales.
            // NOTE: The language, country and variant strings are interned.
            if (aLocale.getLanguage() == "th" && aLocale.getCountry() == "TH") {
                cal = new BuddhistCalendar(zone, aLocale);
            } else if (aLocale.getVariant() == "JP" && aLocale.getLanguage() == "ja"
                       && aLocale.getCountry() == "JP") {
                cal = new JapaneseImperialCalendar(zone, aLocale);
            } else {
                cal = new GregorianCalendar(zone, aLocale);
            }
        }
        return cal;
    }
}


📢 四、总结Tips


工厂模式的意义:将实例化对象的代码提取出来,放到一个类中统一管理和维护,达到和主项目的依赖关系的解耦。从而提高项目的扩展和维护性。

三种工厂模式 (简单工厂模式、工厂方法模式、抽象工厂模式)

设计模式的依赖抽象原则

创建对象实例时,不要直接 new 类, 而是把这个new 类的动作放在一个工厂的方法中,并返回。有的书上说,变量不要直接持有具体类的引用。

不要让类继承具体类,而是继承抽象类或者是实现interface(接口)、不要覆盖基类中已经实现的方法


相关文章
|
4天前
|
数据采集 人工智能 Java
Java产科专科电子病历系统源码
产科专科电子病历系统,全结构化设计,实现产科专科电子病历与院内HIS、LIS、PACS信息系统、区域妇幼信息平台的三级互联互通,系统由门诊系统、住院系统、数据统计模块三部分组成,它管理了孕妇从怀孕开始到生产结束42天一系列医院保健服务信息。
19 4
|
11天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
39 2
|
15天前
|
人工智能 监控 数据可视化
Java智慧工地信息管理平台源码 智慧工地信息化解决方案SaaS源码 支持二次开发
智慧工地系统是依托物联网、互联网、AI、可视化建立的大数据管理平台,是一种全新的管理模式,能够实现劳务管理、安全施工、绿色施工的智能化和互联网化。围绕施工现场管理的人、机、料、法、环五大维度,以及施工过程管理的进度、质量、安全三大体系为基础应用,实现全面高效的工程管理需求,满足工地多角色、多视角的有效监管,实现工程建设管理的降本增效,为监管平台提供数据支撑。
32 3
|
20天前
|
运维 自然语言处理 供应链
Java云HIS医院管理系统源码 病案管理、医保业务、门诊、住院、电子病历编辑器
通过门诊的申请,或者直接住院登记,通过”护士工作站“分配患者,完成后,进入医生患者列表,医生对应开具”长期医嘱“和”临时医嘱“,并在电子病历中,记录病情。病人出院时,停止长期医嘱,开具出院医嘱。进入出院审核,审核医嘱与住院通过后,病人结清缴费,完成出院。
58 3
|
22天前
|
jenkins Java 测试技术
如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例详细说明
本文介绍了如何使用 Jenkins 自动发布 Java 代码,通过一个电商公司后端服务的实际案例,详细说明了从 Jenkins 安装配置到自动构建、测试和部署的全流程。文中还提供了一个 Jenkinsfile 示例,并分享了实践经验,强调了版本控制、自动化测试等关键点的重要性。
58 3
|
24天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
45 2
|
25天前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
14 2
|
26天前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
29天前
|
移动开发 前端开发 JavaScript
java家政系统成品源码的关键特点和技术应用
家政系统成品源码是已开发完成的家政服务管理软件,支持用户注册、登录、管理个人资料,家政人员信息管理,服务项目分类,订单与预约管理,支付集成,评价与反馈,地图定位等功能。适用于各种规模的家政服务公司,采用uniapp、SpringBoot、MySQL等技术栈,确保高效管理和优质用户体验。
|
30天前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
19 1
下一篇
无影云桌面