0

0

什么是Java中的组合优于继承原则_降低类耦合度的设计模式实践

P粉602998670

P粉602998670

发布时间:2026-03-16 14:34:03

|

664人浏览过

|

来源于php中文网

原创

该用 private final 字段+构造器注入组合对象:当复用功能但非“是一种”关系时,可提前校验依赖、便于测试、避免 new 硬编码和字段暴露;继承 Vector 实现 Stack 是反例,因语义混乱、无法替换底层、易受父类变更影响。

什么是java中的组合优于继承原则_降低类耦合度的设计模式实践

什么时候该用 private final 字段 + 构造器注入组合对象

当你要复用另一个类的功能,但两者之间不是“是一种”的关系(比如 Car 不是 Engine 的一种),就该用组合——而且首选 private final 字段配合构造器注入。

这样做能提前拦截空依赖、避免运行时 NullPointerException,也方便单元测试时传入 MockEngineStubTransmission

  • 别在类里直接写 new Engine():无法替换、无法 mock、生命周期失控
  • 别用 publicprotected 字段暴露组合对象:破坏封装,外部可能篡改状态
  • 别提供 setEngine() 除非真需要运行时热切换(比如灰度切换支付渠道)
  • 构造器参数务必用 Objects.requireNonNull() 校验:失败早于业务逻辑,不等到 start() 才崩

为什么继承 Vector 实现 Stack 是个经典反例

JDK 早期的 Stack 继承自 Vector,结果它既暴露了 Vector 全套增删查改接口(add()remove()get()),又靠重写部分方法来模拟栈行为——这导致语义混乱、线程不安全、且无法替换底层容器。

正确做法是组合一个 List,只暴露 push()pop(),内部委托调用:

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

社研通
社研通

文科研究生的学术加速器

下载
public class Stack<E> {
    private final List<E> delegate = new ArrayList<>();
    public void push(E item) { delegate.add(item); }
    public E pop() { return delegate.remove(delegate.size() - 1); }
}
  • 继承让 Stack 意外获得 Vector 的同步开销和扩容策略,而实际不需要
  • 子类无法绕过父类构造器链,但组合对象可随时换成 LinkedList 或带缓存的自定义实现
  • 如果父类某天删掉 ensureCapacity() 这种 protected 方法,所有继承它的子类都会编译失败

哪些继承场景其实很危险,但开发者常踩坑

不是所有“看起来像父子关系”的地方都适合继承。尤其当父类没明确声明“可被继承”时,强行 extends 很容易引发隐性崩溃。

  • HashSet 重写了 add() 去操作内部 HashMap,你再继承它并重写 add(),很可能漏掉 map.put(),导致 size() 和真实元素数不一致
  • StringLocalDateTime 被设为 final,不是为了“防扩展”,而是因为它们的状态契约太脆弱,子类根本无法维持不变量
  • 父类 JavaDoc 里没写 “This class is designed for inheritance”,却有大量 private 实现细节或 final 方法——这时继承基本等于埋雷
  • 子类重写父类方法后抛出 IllegalArgumentException,但父类文档没说明这个约束,调用方按原契约使用就出错

组合不是拒绝继承,而是让继承回归语义本质

Java 里依然需要继承:比如 AbstractList 提供骨架实现,HttpServlet 定义统一回调入口,Spring 的 ApplicationRunner 强制框架约定。但这些继承必须窄而深——只解决“类型契约”,不掺杂“能力装配”。

一个类可以既有继承又有组合,关键看分工是否清晰:

  • 继承负责“我是谁”(class JsonLogger extends Logger 表明它是 Logger 的一种)
  • 组合负责“我用什么”(JsonLogger 内部持有 private final ObjectMapper 来序列化)
  • 如果发现子类要覆盖父类大部分方法,或者总在补救父类设计缺陷(比如加 if (isLegacyMode)),那大概率本该用组合

最常被忽略的一点是:组合的灵活性不来自“多写几行代码”,而来自它把变化点显式地拎出来——哪个对象可换、何时换、怎么换,全都落在构造器参数或 setter 上,而不是藏在继承树深处。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

161

2025.08.06

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

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

89

2026.01.26

string转int
string转int

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

1071

2023.08.02

if什么意思
if什么意思

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

847

2023.08.22

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1999

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

681

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2418

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

49

2026.01.19

C++多线程并发控制与线程安全设计实践
C++多线程并发控制与线程安全设计实践

本专题围绕 C++ 在高性能系统开发中的并发控制技术展开,系统讲解多线程编程模型与线程安全设计方法。内容包括互斥锁、读写锁、条件变量、原子操作以及线程池实现机制,同时结合实际案例分析并发竞争、死锁避免与性能优化策略。通过实践讲解,帮助开发者掌握构建稳定高效并发系统的关键技术。

4

2026.03.16

热门下载

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

精品课程

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

共23课时 | 4.5万人学习

C# 教程
C# 教程

共94课时 | 11.5万人学习

Java 教程
Java 教程

共578课时 | 83.2万人学习

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

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