0

0

如何在Java中通过组合替代复杂的继承树_Favor Composition over Inheritance

P粉602998670

P粉602998670

发布时间:2026-02-16 11:19:03

|

645人浏览过

|

来源于php中文网

原创

继承树过深导致难改难测,因子类对父类隐式依赖强;组合通过抽接口、提字段、委托调用解耦,但需权衡is-a语义、类型兼容与性能等场景。

如何在java中通过组合替代复杂的继承树_favor composition over inheritance

为什么继承树一深就难改又难测

继承关系越深,子类对父类的隐式依赖就越强。改一个 protected 方法,可能让下游十几个子类行为突变;加个新子类,得反复确认它有没有无意覆盖了关键钩子方法;写单元测试时,还得把整个父类初始化链拉起来——哪怕你只关心子类里一行逻辑。

组合能切断这种“牵一发而动全身”的耦合。把功能拆成独立、可替换的部件,用字段持有它们,而不是靠 extends 继承层级来拼凑行为。

常见错误现象:NullPointerException 频繁出现在父类构造器里调用被子类重写的 hook 方法(因子类字段尚未初始化);final 类无法扩展,但已有继承体系又不允许重构;mock 测试时发现不得不 mock 整个父类链。

怎么把一个继承结构快速转成组合

核心动作就三步:抽接口、提字段、委托调用。不追求一步到位,先从最常变、最易错的那块逻辑下手。

立即学习Java免费学习笔记(深入)”;

  • 找出继承树中变化最频繁的职责(比如日志策略、序列化方式、重试逻辑),把它定义为接口,如 RetryPolicySerializer
  • 在原基类或目标类中,删掉 extends,新增字段,如 private final RetryPolicy retryPolicy
  • 把原来由父类实现、子类定制的方法,改成调用该字段,如 retryPolicy.shouldRetry(exception)
  • 构造器里传入具体实现,支持运行时替换(别用 new DefaultRetryPolicy() 硬编码)

示例:原 HttpClient 继承 BaseClient,再继承 NetworkClient;现在直接在 HttpClient 里持有一个 NetworkLayer 字段,所有网络调用都委托给它。

芦笋演示
芦笋演示

一键出成片的录屏演示软件,专为制作产品演示、教学课程和使用教程而设计。

下载

组合后怎么处理“is-a”语义和类型兼容问题

继承天然表达 is-a,组合则更倾向 has-a。但业务代码里常需要统一类型处理,比如 List<client></client> 里混放不同客户端。

解决路径很实际:

  • 保留顶层接口(如 Client),让所有组合类实现它,而非继承某个抽象类
  • 避免在接口里塞太多方法;按场景拆小接口,如 SyncClientAsyncClient,组合类按需实现
  • 如果旧代码强依赖继承体系(比如 Spring 的 @Primary Bean 冲突),可用包装器模式:新建一个 LegacyClientWrapper 实现原抽象类,内部持组合对象并转发调用
  • 注意泛型擦除下 Class.isAssignableFrom() 不再成立,运行时类型判断要改用字段/接口检查

哪些情况别硬套组合,小心过早抽象

不是所有继承都该被消灭。组合带来灵活性的同时,也增加间接层和对象数量。

以下情况保持继承更合理:

  • 类之间确实是严格的 is-a 关系,且行为差异仅在于少量参数(如 ArrayListLinkedList 都是 List,但底层数据结构不可互换)
  • 框架强制要求继承(如 Android 的 Activity、JUnit 的 TestCase),此时组合只能作为内部优化手段,外部仍需继承
  • 性能敏感路径,频繁委托调用引发可观开销(如高频循环里的 formatter.format()),这时内联或模板方法更合适

最容易被忽略的一点:组合不等于“每个行为都做成接口+实现类”。很多逻辑其实只需一个配置枚举或函数式接口参数,比如 Comparator<t></t>Consumer<string></string> —— 它们比定义十个 LogStrategy 子类轻量得多。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

134

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

81

2026.01.26

软件测试常用工具
软件测试常用工具

软件测试常用工具有Selenium、JUnit、Appium、JMeter、LoadRunner、Postman、TestNG、LoadUI、SoapUI、Cucumber和Robot Framework等等。测试人员可以根据具体的测试需求和技术栈选择适合的工具,提高测试效率和准确性 。

448

2023.10.13

java测试工具有哪些
java测试工具有哪些

java测试工具有JUnit、TestNG、Mockito、Selenium、Apache JMeter和Cucumber。php还给大家带来了java有关的教程,欢迎大家前来学习阅读,希望对大家能有所帮助。

309

2023.10.23

Java 单元测试
Java 单元测试

本专题聚焦 Java 在软件测试与持续集成流程中的实战应用,系统讲解 JUnit 单元测试框架、Mock 数据、集成测试、代码覆盖率分析、Maven 测试配置、CI/CD 流水线搭建(Jenkins、GitHub Actions)等关键内容。通过实战案例(如企业级项目自动化测试、持续交付流程搭建),帮助学习者掌握 Java 项目质量保障与自动化交付的完整体系。

22

2025.10.24

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

145

2026.02.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.6万人学习

C# 教程
C# 教程

共94课时 | 9.5万人学习

Java 教程
Java 教程

共578课时 | 66.1万人学习

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

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