0

0

如何在Java中使用组合优于继承原则_降低类耦合度的设计技巧

P粉602998670

P粉602998670

发布时间:2026-03-15 12:30:31

|

842人浏览过

|

来源于php中文网

原创

该用组合而非继承当子类仅为复用代码且不满足“is-a”关系时,如Car与Engine;常见错误包括ClassCastException、空方法重写、父类修改引发崩溃;组合通过委托、接口和依赖注入实现多态与策略切换,更安全可控。

如何在java中使用组合优于继承原则_降低类耦合度的设计技巧

什么时候该用组合而不是继承

继承不是不能用,但一旦子类只为了复用代码而存在,又不真正符合“is-a”关系,就该立刻警觉。比如 EngineCarCar 有发动机,不是“是一种发动机”,所以不该让 Car extends Engine

常见错误现象:ClassCastException、子类被迫重写大量空方法、修改父类导致下游意外崩溃。

  • 当父类行为不稳定(比如频繁改 protected 方法逻辑),用组合能隔离变化
  • 需要运行时切换行为(比如换不同 PaymentStrategy),继承做不到,组合配合接口可以
  • 想测试某个功能模块,继承会让测试必须搭起整个类层次,组合只需 mock 掉依赖对象

怎么把继承改成组合:三步替换法

核心是把“继承来的字段和方法”转成“持有的对象 + 委托调用”。不是简单加个字段就完事,重点在控制权转移。

假设原代码是 class AdminUser extends User,现在要重构:

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

  • 删掉 extends User,声明私有字段 private User user;
  • 在构造器里初始化 user = new User(...);,或通过参数传入(利于测试)
  • 把原来直接调用的 getName()getRole() 等方法,改成 return user.getName(); —— 这叫委托(delegation),不是透传

注意:别偷懒写 public User getUser() { return user; },这等于把封装捅了个洞,外部能绕过你的控制逻辑直接操作 user

组合中如何处理多态和策略切换

继承天然支持多态,组合得靠接口+依赖注入。关键不是“能不能”,而是“谁负责决定用哪个实现”。

Face++旷视
Face++旷视

Face⁺⁺ AI开放平台

下载

比如日志模块,原来用 FileLogger extends LoggerDbLogger extends Logger,现在改为:

interface LogSink {
    void write(String msg);
}
class FileLogSink implements LogSink { ... }
class ConsoleLogSink implements LogSink { ... }
class Service {
    private final LogSink sink;
    Service(LogSink sink) { this.sink = sink; } // 运行时决定
}

性能影响很小,反而是更可控:避免继承树过深带来的方法解析开销;兼容性也更好——旧代码可继续用继承版,新模块统一走组合接口,逐步迁移。

容易踩的坑:Service 如果自己 new FileLogSink,就又写死了;必须由上层(比如 Spring 容器或工厂类)注入,否则组合就退化成硬编码。

为什么 IDE 不会帮你自动完成这个重构

因为组合不是语法转换,而是语义重构。IDE 可以把 extends 删掉、加字段、生成委托方法,但它没法判断:User 的哪些方法该暴露,哪些该拦截加工,哪些该拒绝调用。

真正难的是设计决策点:

  • AdminUser 调用 user.delete() 是否合法?继承下无法阻止,组合下你可以在委托前加校验
  • 是否需要对 user.getName() 返回值做脱敏?组合让你有一层自由加工的空间
  • 多个组合对象之间要不要共享状态?比如 CacheManagerMetricsCollector 是否共用同一个 Timer 实例?继承完全没这个视角

这些都得人来想清楚边界。工具只能搬砖,砌墙的方向得自己定。

热门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

java多态详细介绍
java多态详细介绍

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

27

2025.11.27

java多态详细介绍
java多态详细介绍

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

27

2025.11.27

java多态详细介绍
java多态详细介绍

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

27

2025.11.27

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

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

1974

2023.10.19

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

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

659

2025.10.17

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

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

2406

2025.12.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 82.1万人学习

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

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