如何通过多态实现工厂模式

发布时间 - 2026-02-02 00:00:00    点击率:
多态是工厂模式的底层支撑,工厂返回共同接口(如ABC或interface)的实例,各子类实现同名方法,调用方只依赖接口而不感知具体类型,否则破坏多态契约。

多态是工厂模式的底层支撑,不是可选技巧

工厂模式本身不强制要求多态,但一旦涉及“创建不同子类实例并统一调用同名方法”,就必须依赖多态。否则你得写一堆 if-elif 判断类型再手动调用对应方法,既难维护又违背开闭原则。

关键在于:工厂返回的不是具体类型,而是共同接口(如抽象基类或协议),所有产品类实现该接口——这才是多态生效的前提。

Python 中用 ABC + @abstractmethod 强制多态契约

没有编译期检查时,靠文档或约定容易漏实现。用 abc.ABC@abstractmethod 能在实例化时立刻报错,避免运行时才发现某个子类没实现 execute() 这类关键方法。

  • Product 必须声明为 ABC,且核心方法标为 @abstractmethod
  • 每个具体产品类(如 FileExporterApiExporter)必须继承它并实现全部抽象方法
  • 工厂函数返回类型注解应为 Product,而非具体子类,否则静态检查工具(如 mypy)会警告类型不匹配
from abc import ABC, abstractmethod

class Product(ABC):
    @abstractmethod
    def export(self) -> str:
        ...

class FileExporter(Product):
    def export(self) -> str:
        return "saved to file"

class ApiExporter(Product):
    def export(self) -> str:
        return "sent via HTTP"

def create_exporter(export_type: str) -> Product:
    if export_type == "file":
        return FileExporter()
    elif export_type == "api":
        return ApiExporter()
    else:
        raise ValueError(f"Unknown type: {export_type}")

Java 里 interface 比 abstract class 更适合做产品契约

工厂模式中产品之间往往无共用字段或逻辑,只共享行为签名。用 interface 更轻量,也避免单继承限制——比如某个导出器既要实现 Exporter,又要继承 RetryableTask,用 abstract class 就卡死了。

  • 定义 Exporter 接口,只含 export() 方法
  • 各实现类用 implements Exporter,不强求继承同一父类
  • 工厂方法返回类型写成 Exporter,调用方只依赖接口,完全不知道背后是 FileExporter 还是 CloudExporter
  • 注意:Java 8+ 可在 interface 中加 default 方法提供通用逻辑,但别滥用——工厂模式里“通用”逻辑通常属于工厂本身或策略类,不该塞进产品接口

JavaScript 的鸭子类型让多态更隐蔽,也更容易出错

JS 没有编译期类型约束,所谓“多态”全靠运行时对象恰好有 process() 方法。这导致两个典型问题:拼错方法名、漏传参数,到真正调用时才暴露。

  • 工厂返回的对象必须确保有统一的方法名和参数结构,建议用 Object.assign() 或构造函数校验字段
  • 在工厂内部加简单断言,例如:if (typeof instance.process !== 'function') throw new Error('Missing process method')
  • 配合 TypeScript 是更稳妥的选择:定义 interface Exporter,工厂函数签名写成 createExporter(type: string): Exporter,编辑器能实时提示错误
interface Exporter {
  export(): strin

g; } function createExporter(type: string): Exporter { switch (type) { case 'file': return { export: () => 'saved to file' }; case 'api': return { export: () => 'sent via HTTP' }; default: throw new Error(`Unknown exporter: ${type}`); } }
多态本身不难,难的是在工厂返回后,调用方是否真的只依赖接口、完全不感知具体类型。只要出现 instanceof 判断或类型断言,就说明多态契约已被破坏,工厂模式的价值也就打折扣了。


# javascript  # python  # java  # js  # typescript  # 工具  # ai  # switch  # elif  # String  # Object  # if  # 多态  # 父类  # 子类  # 构造函数  # throw  # Error  # 继承  # 接口  #   # class  # Interface 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 高防服务器租用如何选择配置与防御等级?  高端智能建站公司优选:品牌定制与SEO优化一站式服务  Python文本处理实践_日志清洗解析【指导】  怎么用AI帮你为初创公司进行市场定位分析?  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  Laravel如何配置和使用缓存?(Redis代码示例)  android nfc常用标签读取总结  中山网站制作网页,中山新生登记系统登记流程?  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  高配服务器限时抢购:企业级配置与回收服务一站式优惠方案  做企业网站制作流程,企业网站制作基本流程有哪些?  如何快速生成凡客建站的专业级图册?  网站制作报价单模板图片,小松挖机官方网站报价?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  制作公司内部网站有哪些,内网如何建网站?  Claude怎样写结构化提示词_Claude结构化提示词写法【教程】  如何获取免费开源的自助建站系统源码?  Python面向对象测试方法_mock解析【教程】  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  中国移动官方网站首页入口 中国移动官网网页登录  Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言  如何用已有域名快速搭建网站?  JS去除重复并统计数量的实现方法  ,交易猫的商品怎么发布到网站上去?  音乐网站服务器如何优化API响应速度?  小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像  Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  网站制作价目表怎么做,珍爱网婚介费用多少?  Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法  Laravel如何升级到最新版本?(升级指南和步骤)  HTML 中如何正确使用模板变量为元素的 name 属性赋值  javascript基本数据类型及类型检测常用方法小结  利用 Google AI 进行 YouTube 视频 SEO 描述优化  如何在橙子建站上传落地页?操作指南详解  详解阿里云nginx服务器多站点的配置  Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案  如何快速使用云服务器搭建个人网站?  Laravel怎么防止CSRF攻击_Laravel CSRF保护中间件原理与实践  历史网站制作软件,华为如何找回被删除的网站?  如何在景安服务器上快速搭建个人网站?  Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全  如何快速搭建虚拟主机网站?新手必看指南  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  三星、SK海力士获美批准:可向中国出口芯片制造设备  如何确认建站备案号应放置的具体位置?  标题:Vue + Vuex 项目中正确使用 JWT 进行身份认证的实践指南  Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】  如何确保西部建站助手FTP传输的安全性?