0

0

开闭原则 (OCP)_如何在不修改源码的情况下扩展功能

P粉602998670

P粉602998670

发布时间:2026-02-25 17:42:10

|

447人浏览过

|

来源于php中文网

原创

开闭原则是“对扩展开放、对修改关闭”,核心是不破坏已有逻辑的实现;可通过新增类、接口、注册表或依赖注入实现扩展,而非修改原有代码。

开闭原则 (ocp)_如何在不修改源码的情况下扩展功能

开闭原则不是“不改代码”,而是“不改已有逻辑的实现”

开闭原则(OCP)常被误解为“绝对不能动源码”,其实质是:对扩展开放,对修改关闭——重点在「已有行为逻辑不被破坏」。你完全可以新增文件、新写类、新注册函数,只要老代码调用路径不变、契约(接口/参数/返回值)不变,就算守住 OCP。

最容易踩的坑是强行塞 if-else 或 switch 到原有函数里加新分支,比如在 processOrder() 里加 if type === 'vip' {...}。这看似没动“结构”,但每次加类型都要改它,违反了“修改关闭”。

  • 正确做法:把订单处理逻辑抽成策略接口,OrderProcessor 只负责调度,具体实现由 VipOrderHandlerNormalOrderHandler 等独立类承担
  • 关键判断点:改动后,单元测试中所有原有 processOrder() 的测试用例是否仍全部通过?如果需要重写或跳过某些断言,说明已破坏封闭性
  • 语言差异:TypeScript 可靠接口 + 工厂函数;Python 常用 abc.ABC + 注册表;Go 用接口+依赖注入,避免直接 new 具体类型

如何识别“该走扩展而不是修改”的信号

不是所有新增需求都适合 OCP,但以下现象出现时,强烈建议停手、重构、再扩展:

  • 你在改一个已有函数,只为支持一种新输入类型,而它原本只处理一种 —— 比如 parseJSON(data) 突然要兼容 parseXML(data)
  • 你发现要加大量 typeofinstanceofdata.format === 'yaml' 分支判断
  • 同事在 Code Review 里问:“这个 if 是临时的吗?”——大概率不是临时的,是扩展点缺失的征兆
  • CI 测试里某个模块的覆盖率突然掉了一截,因为新加逻辑绕过了原有路径

这时候,与其补丁式修改,不如定义统一入口(如 Parser 接口),让 JSONParserXMLParser 各自实现,由工厂或配置决定用哪个。

盛世企业网站管理系统1.1.2
盛世企业网站管理系统1.1.2

免费 盛世企业网站管理系统(SnSee)系统完全免费使用,无任何功能模块使用限制,在使用过程中如遇到相关问题可以去官方论坛参与讨论。开源 系统Web代码完全开源,在您使用过程中可以根据自已实际情况加以调整或修改,完全可以满足您的需求。强大且灵活 独创的多语言功能,可以直接在后台自由设定语言版本,其语言版本不限数量,可根据自已需要进行任意设置;系统各模块可在后台自由设置及开启;强大且适用的后台管理支

下载

依赖注入和注册表是落地 OCP 最常用的两个杠杆

没有机制支撑,OCP 就是空谈。硬编码 new 实例、全局变量挂处理器、手动改 switch 表,都会让扩展变成高风险操作。

  • 用依赖注入:把具体实现作为参数传入,而非在类内部 new —— new OrderService(new VipOrderHandler()),替换 handler 不用碰 OrderService 源码
  • 用注册表:启动时动态注册处理器,比如 ParserRegistry.register('yaml', new YamlParser()),后续 ParserRegistry.get('yaml').parse(...) 自动路由,增删格式都不动核心调度逻辑
  • 警惕陷阱:注册表本身若用静态变量 + 隐式初始化(如 import 时就执行 register),会导致测试间污染;应显式初始化,或用容器生命周期管理
  • 性能提示:注册表查找一般 O(1),但若用字符串匹配做路由(如正则 or includes),可能引入隐式性能衰减,优先用精确 key 查找

测试覆盖是验证 OCP 是否真被守住的唯一标尺

写完扩展逻辑后,运行原有全部测试。如果失败,要么是契约被破坏(比如新 handler 返回了不同结构),要么是旧路径被意外干扰(比如注册表覆盖了默认实现)。

  • 确保每个扩展类都有对应单元测试,但更重要的是:跑一遍老模块的完整测试集,尤其是集成场景(如 processOrder() 调用链)
  • 不要只测“新功能能用”,要测“老功能没变”——比如原来 processOrder({type: 'normal'}) 返回 {status: 'ok'},现在还得一样
  • Mock 依赖时注意粒度:若 mock 掉整个 PaymentService,可能掩盖新 handler 对它的新调用方式;应保留真实协作流,只隔离外部副作用(如网络、DB)

OCP 的复杂点不在概念,而在边界感——什么时候该划新接口,什么时候该复用旧字段,哪些契约一旦定下就不能松动。这些判断没法靠规则穷举,得靠每次改代码时多问一句:“我动的这一行,会让谁的测试崩掉?”

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

40

2026.02.13

TypeScript全栈项目架构与接口规范设计
TypeScript全栈项目架构与接口规范设计

本专题面向全栈开发者,系统讲解基于 TypeScript 构建前后端统一技术栈的工程化实践。内容涵盖项目分层设计、接口协议规范、类型共享机制、错误码体系设计、接口自动化生成与文档维护方案。通过完整项目示例,帮助开发者构建结构清晰、类型安全、易维护的现代全栈应用架构。

15

2026.02.25

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

830

2023.08.22

switch语句用法
switch语句用法

switch语句用法:1、Switch语句只能用于整数类型,枚举类型和String类型,不能用于浮点数类型和布尔类型;2、每个case语句后面必须跟着一个break语句,以防止执行其他case的代码块,没有break语句,将会继续执行下一个case的代码块;3、可以在一个case语句中匹配多个值,使用逗号分隔;4、Switch语句中的default代码块是可选的等等。

562

2023.09.21

Java switch的用法
Java switch的用法

Java中的switch语句用于根据不同的条件执行不同的代码块。想了解更多switch的相关内容,可以阅读本专题下面的文章。

435

2024.03.13

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

866

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

452

2024.06.27

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

87

2025.09.18

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

127

2026.02.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号