23种设计模式-创建型模式-工厂方法

news2025/3/27 17:08:19

文章目录

  • 简介
  • 场景
  • 问题
    • 1. 直接依赖具体实现
    • 2. 违反开闭原则
    • 3. 条件分支泛滥
    • 4. 代码重复风险
  • 解决
    • 根本问题
    • 完整类图
    • 完整代码
    • 说明
    • 核心优势
    • 代码优化
      • 静态配置表
      • 动态策略
  • 总结

简介

工厂方法是一种创建型设计模式,它提供了在父类中创建对象的接口,但允许子类改变将要创建的对象的类型。工厂方法(Factory Method)的核心目标是解耦对象创建与使用逻辑。

场景

假设你正在创建一个物流管理应用程序。物流只采用卡车运输,所以当前只能处理卡车运输,大部分代码都位于 Truck 类中。

/** 客户端运输管理类 */
class LogisticsManager {
    public void transportCargo() {
        // 直接依赖具体类 Truck
        Truck truck = new Truck();
        truck.deliver();
        // 新增其他运输类型需要继续扩展条件分支
    }
}

/** 卡车类(硬编码具体实现) */
class Truck {
    public void deliver() {
        System.out.println("陆地运输:卡车配送");
    }
}

一段时间后,你想把海运物流也加到应用中。
这个时候你会发现,你可能需要再创建一个Ship类用于处理海运的请求,并且还需要修改主流程代码。你可能会做出如下修改:

/** 客户端运输管理类 */
class LogisticsManager {
    public void transportCargo(String type) {
        // 直接依赖具体类 Truck
        if ("land".equals(type)) {
            Truck truck = new Truck();
            truck.deliver();
        } else if ("sea".equals(type)) {
            Ship ship = new Ship(); // 新增 Ship 需修改此处
            ship.deliver();
        }
        // 新增其他运输类型需要继续扩展条件分支
    }
}

/** 卡车类(硬编码具体实现) */
class Truck {
    public void deliver() {
        System.out.println("陆地运输:卡车配送");
    }
}

/** 新增船舶类时被迫修改既有代码 */
class Ship {
    public void deliver() {
        System.out.println("海运:船舶配送");
    }
}

问题

1. 直接依赖具体实现

LogisticsManager 直接通过 new Truck() 硬编码创建对象,客户端代码与 Truck 类强耦合。新增 Ship 类时必须修改 transportCargo 方法的逻辑。

2. 违反开闭原则

每添加一种新的运输工具(例如 Ship 或 Airplane),都需要:

  • 修改 transportCargo 方法,增加条件分支
  • 引入对其他具体类的依赖
  • 导致核心业务逻辑频繁变更

3. 条件分支泛滥

if-else 或 switch 语句扩散到多个位置(例如支付方式、日志记录等其他模块重复出现类似代码),维护成本指数级增长。

4. 代码重复风险

若其他模块(如计费系统)也需要根据运输类型创建对象,必须重复编写相同的条件分支逻辑,系统复杂度失控。

解决

根本问题

对象创建逻辑与业务逻辑深度耦合,未遵循「面向接口编程」原则 。工厂方法模式通过把创建过程封装到独立层级(如新增 LandLogistics / SeaLogistics 子类)可彻底解决这些问题。

我们用工厂方法代替直接调用对象构造函数(也就是使用new)。但是对象还是通过new创建,只是new变成了在工厂方法里调用。工厂方法返回的对象就是“产品”。

子类可以修改工厂方法返回的对象类型

粗略看着这种更改可能没啥意义,我们只是改变了程序中调用构造函数的位置而已。但是,仔细想一下,现在你可以在子类中重写工厂方法,从而改变它创建产品的类型。
但有一点需要注意:只有这些产品具有共同的基类或者接口时,子类才能返回不同类型的产品,同时基类中工厂方法的返回类型也要声明成这个共有接口。

所有产品都必须使用同一接口
举例来说,卡车Truck和轮船Ship类都必须实现运输Trans­port接口,这个接口声明了一个deliver方法。每个类都会以不同的方式实现这个方法:卡车走陆路交付货物,轮船走海路交付货物。陆路运输Road­Logis­tics类里的工厂方法返回卡车对象,而海路运输Sea­Logis­tics类就返回轮船对象。

完整类图

在这里插入图片描述

完整代码

// 抽象产品接口:所有运输工具必须实现此接口 [^6]
interface Transport {
    void deliver();
}

// 具体产品实现:卡车类
class Truck implements Transport {
    @Override
    public void deliver() {
        System.out.println("陆地运输:卡车配送");
    }
}

// 具体产品实现:船舶类
class Ship implements Transport {
    @Override
    public void deliver() { 
        System.out.println("海运:船舶配送");
    }
}

// 抽象工厂类(核心工厂方法) [^6]
abstract class Logistics {
    // 模板方法:组合核心逻辑与工厂方法
    public void planDelivery() {
        Transport transport = createTransport();
        transport.deliver();
    }

    // 工厂方法声明:延迟具体运输工具创建到子类
    public abstract Transport createTransport();
}

// 具体工厂类:陆地物流 [^6]
class RoadLogistics extends Logistics {
    @Override
    public Transport createTransport() {
        return new Truck(); // 无需判读条件的直接构造
    }
}

// 具体工厂类:海洋物流
class SeaLogistics extends Logistics {
    @Override
    public Transport createTransport() {
        return new Ship();
    }
}

// 客户端调用示例
class Client {
    public void clientMethod(String logisticsType) {
        Logistics logistics;
        
        // 通过简单映射选择具体工厂
        if ("road".equals(logisticsType)) {
            logistics = new RoadLogistics();
        } else {
            logistics = new SeaLogistics(); 
        }

        // 统一调用接口(核心业务逻辑无需修改)
        logistics.planDelivery();
    }
}

说明

  1. 实现关系:Truck/Ship 实现 Transport 接口
  2. 继承关系:RoadLogistics/SeaLogistics 继承 Logistics 抽象类
  3. 依赖关系:Logistics 依赖 Transport 接口(虚线箭头)

核心优势

  1. 快速扩展能力:新增空运只需添加 AirLogistics 类和 Plane 产品类,零/少修改现有代码
  2. 核心业务稳定:planDelivery() 模板方法与运输工具实现完全解耦
  3. 统一接口调用:客户端通过 Logistics 基类操作所有物流类型,符合里氏替换原则(在使用继承的过程中,子类可以完全替换掉父类,并且软件的功能不受到影响。这个原则是保证代码的健壮性和可扩展性的重要原则之一)

代码优化

以上代码在客户端中还是存在条件判断分支,可以通过静态配置注册表或者动态策略解决。如果新增运输工具,客户端就只需要修改配置文件即可,真正实现零修改现有代码。

静态配置表

// 创建对象注册表
class LogisticsRegistry {
    private static Map<String, Supplier<Logistics>> registry = new HashMap<>();
    
    static {
        registry.put("road", RoadLogistics::new); // 配置化绑定, 可以从配置文件中加载
        registry.put("sea", SeaLogistics::new);
    }
    
    public static Logistics getLogistics(String type) {
        return registry.getOrDefault(type, SeaLogistics::new).get();
    }
}

// 客户端无需任何条件判断
class Client {
    public void clientMethod(String logisticsType) {
        Logistics logistics = LogisticsRegistry.getLogistics(logisticsType);
        logistics.planDelivery();
    }
}

动态策略

若系统存在动态运行时判定逻辑(如根据GPS坐标实时选择最优运输方式),则需在简单工厂层用策略模式处理:

class DynamicLogisticsSelector {
    public Logistics selectByConditions(Context ctx) {
        // 动态策略分支(仍可避免硬编码if)
        return strategyMap.get(ctx.analyze()).create();
    }
}

总结

在这里插入图片描述

  1. 产品(Prod­uct)接口。对于所有由创建者(Creator)以及它的子类构建的对象,这些接口都是通用的。
  2. 具体产品(Con­crete Prod­ucts)是产品接口的不同实现。
  3. 创建者(Cre­ator)声明了返回产品对象的工厂方法。方法的返回对象类型必须与产品接口一致。你可以把工厂方法声明为抽象方法,强制要求每个子类重新实现。或者,你也可以在基类工厂方法中返回默认产品类型。
  4. 具体创建者(Con­crete Cre­ators)会重写基础的工厂方法,让他返回不同类型的产品。注意,并不一定每次调用工厂方法都会创建新的实例。工厂方法也可以返回缓存、对象池或其他来源的已有对象

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2321792.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

142. 环形链表 II——考察数学,难!

142. 环形链表 IIhttps://leetcode.cn/problems/linked-list-cycle-ii/ 给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,…

从零开始:使用Luatools工具高效烧录Air780EPM核心板项目的完整指南

本文将深入讲解如何使用Luatools工具烧录一个具体的项目到Air780EPM开发板中。如何使用官方推荐的Luatools工具&#xff08;一款跨平台、命令行驱动的烧录利器&#xff09;&#xff0c;通过“环境配置→硬件连接→参数设置→一键烧录”四大步骤&#xff0c;帮助用户实现Air780E…

一套云HIS系统源码,系统融合HIS与EMR,基于云端部署,采用B/S架构与SaaS模式

云HIS系统完全基于云端部署&#xff0c;采用B/S架构&#xff0c;并通过软件即服务&#xff08;SaaS&#xff09;的形式面向二级及以下医院可快速交付、便捷运维、云化的医院核心业务平台产品。融合医院HIS和EMR两大主营系统&#xff0c;构建涵盖患者、费用、医嘱、电子病历等核…

C++数据结构(搜索二叉树)

1.二叉树搜索的概念 二叉搜索数也成为二叉排序树&#xff0c;它或者是一颗空树&#xff0c;或者是满足以下性质的树&#xff1a; 1.若他的左子树不为空&#xff0c;则左子树上的所有节点的值都小于等于根节点的值。 2.若他的右子树不为空&#xff0c;则右子树上的所有节点的值…

OpenCV图像拼接(6)图像拼接模块的用于创建权重图函数createWeightMap()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::detail::createWeightMap 是 OpenCV 库中用于图像拼接模块的一个函数&#xff0c;主要用于创建权重图。这个权重图在图像拼接过程中扮演着重…

炫酷的HTML5粒子动画特效实现详解

炫酷的HTML5粒子动画特效实现详解 这里写目录标题 炫酷的HTML5粒子动画特效实现详解项目介绍技术栈项目架构1. HTML结构2. 样式设计 核心实现1. 粒子类设计2. 动画效果实现星空效果烟花效果雨滴效果 3. 鼠标交互 性能优化效果展示总结 项目介绍 本文将详细介绍如何使用HTML5 C…

YoloV8训练和平精英人物检测模型

概述 和平精英人物检测&#xff0c;可以识别游戏中所有人物角色&#xff0c;并通过绘制框将人物选中&#xff0c;训练的模型仅仅具有识别功能&#xff0c;可以识别游戏中的视频、图片等文件&#xff0c;搭配Autox.js可以推理&#xff0c;实现实时绘制&#xff0c;但是对手机性…

BC93 公务员面试

&#x1f680;个人主页&#xff1a;BabyZZの秘密日记 &#x1f4d6;收入专栏&#xff1a;C语言练习题分享 &#x1f30d;文章目入 #include <stdio.h> int main() {int score 0, max 0, min 100, sum 0, count 0; while (scanf("%d", &score) ! EOF){…

3.0 Disruptor的使用介绍(一)

Disruptor: 其官网定义为&#xff1a;“A High Performance Inter-Thread Messaging Library”&#xff0c;即&#xff1a;线程间的高性能消息框架&#xff0c;与Labview的生产者、消费者模型很相似。 其组成部分比较多&#xff0c;先介绍几个常用的概念&#xff1a; …

[深度学习]图像分类项目-食物分类

图像分类项目-食物分类(监督学习和半监督学习) 文章目录 图像分类项目-食物分类(监督学习和半监督学习)项目介绍数据处理设定随机种子读取文件内容图像增广定义Dataset类 模型定义迁移学习 定义超参Adam和AdamW 训练过程半监督学习定义Dataset类模型定义定义超参训练过程 项目介…

java8循环解压zip文件---实现Excel文件数据追加

java8循环追加Excel数据 实际遇到问题&#xff1a;定期获取zip文件&#xff0c;zip文件内有几个固定模板的Excel文件&#xff0c;有的Excel文件可能还包含多个sheet。 有段时间一次性获取到好几个zip包&#xff0c;需要将这些包都解压&#xff0c;并且按照不同的文件名、sheet进…

基于SpringBoot的电影售票系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

SQL Server 2022 安装问题

一、安装与配置问题 1. SQL Server 2022 安装失败怎么办&#xff1f; 常见原因&#xff1a; 硬件或操作系统不满足最低要求&#xff08;如内存、磁盘空间不足&#xff09;。未关闭防火墙或杀毒软件。之前版本的 SQL Server 残留文件未清理。 解决方案&#xff1a; 确保硬件配…

MySQL 8.0.41安装教程(附安装包)mysql8.0.41图文详细安装教程

文章目录 前言一、MySQL 8.0.41下载安装包二、MySQL 8.0.41安装教程1.启动安装程序2.选择安装模式3.选定安装组件4.确认安装设置5.执行安装操作6.安装进行中7.设置数据库密码8.继续点击下一步9.执行配置操作10.完成配置11. 再次点击下一步12.结束安装向导 三、MySQL 8.0.41配置…

深入解析 C++20 中的 std::bind_front:高效函数绑定与参数前置

文章目录 1. 什么是 std::bind_front&#xff1f;2. 使用 std::bind_front2.1 基本用法2.2 绑定多个参数 3. 优势与特点3.1 简化代码3.2 支持可调用对象3.3 支持完美转发 4. 实际应用场景4.1 事件处理4.2 算法通用化4.3 成员函数调用 5. 总结 在现代 C 编程中&#xff0c;函数绑…

python裁剪nc文件数据

问题描述&#xff1a; 若干个nc文件储存全球的1850-2014年月尺度的mrro数据(或其他数据)&#xff0c;从1850-1到2014-12一共1980个月&#xff0c;要提取出最后35年1980.1~2014.12年也就是420个月的数据。 代码实现 def aaa(input_file,output_file,bianliang,start_index,en…

CSS网格布局Grid

目录 一、Grid 网格布局 1.Grid 布局基础 2.网格容器属性 3.网格项目属性 4.高级功能 5.典型应用场景 6.最佳实践 二、Flex和Grid对比 示例&#xff1a; 一、Grid 网格布局 CSS Grid 是一种强大的二维布局系统&#xff0c;能够以行和列的方式精确控制网页布局。它比传…

医院挂号预约小程序|基于微信小程序的医院挂号预约系统设计与实现(源码+数据库+文档)

医院挂号预约小程序 目录 基于微信小程序的医院挂号预约系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、小程序用户端 2、系统服务端 &#xff08;1&#xff09; 用户管理 &#xff08;2&#xff09;医院管理 &#xff08;3&#xff09;医生管理 &#xf…

蓝桥杯第十届 特别的数

题目描述 小明对数位中含有 2、0、1、9 的数字很感兴趣&#xff08;不包括前导 0&#xff09;&#xff0c;在 1 到 40 中这样的数包括 1、2、9、10 至 32、39 和 40&#xff0c;共 28 个&#xff0c;他们的和是 574。 请问&#xff0c;在 1 到 n 中&#xff0c;所有这样的数的…