设计模式学习笔记 - 面向对象 - 7.为什么要多用组合少用继承?如何决定该用组合还是继承?
前言
在面向对象编程中,有一条非常经典的设计原则:组合优于继承,多用组合少用继承。
为什么不推荐使用继承?
组合比继承有哪些优势?
如何判断该用组合还是继承?
为什么不推荐使用继承?
继承是面向对象的四大特性之一,用来表示 is-a 的关系,可以解决代码复用问题。虽然继承有诸多作用,但继承层次太深、过于复杂,会影响到代码的可维护性。所以对弈是否应该在项目中使用继承,有很多争议,很多人觉得继承是一种反模式,应尽量少用。为什么会有这样的争议,我们聚类自来说明下。
假设我们要设计一个关于鸟的类。我们将“鸟类”这样一个抽象的事物概念,定义为一个抽象类 AbstractBird。所有细分的鸟,比如麻雀、鸽子、乌鸦等,都继承这个类。
大部分鸟可以飞,那可以在 AbstractBird 抽象类里面定义一个 fly() 方法吗?答案是否定的。尽管大部分鸟会飞,但也有特例,比如鸵鸟就不会飞。鸵鸟继承具有 fly() 方法的父类,就具有了“飞”这样的行为,这显然不符合我们的对现实世界的认识。当然,你可能会说,我在鸵鸟这个子类中重写 fly() 方法,让它们抛出 UnSupportedMethodException 异常不就可以了吗?代码如下:
public class AbstractBird {// 省略其他属性和方法...public void fly() { /*...*/ }
}public class Ostrich extends AbstractBird {// 省略其他属性和方法...@Overridepublic void fly() {throw new UnSupportedMethodException("I can't fly.");}
}
这种设计思路虽然可以解决问题,但不够优美。因为除了鸵鸟,不会飞的鸟类还有很多,如企鹅。对于这些不会飞的鸟,都要重写 fly() 方法,抛出异常。这样的设计,一方面增加了编码的工作量;另一方面,也违背了我们之后要讲解的最小知识原则(最少知识原则,也叫迪米特法则),暴露不该暴露的接口给外部,增加了类适用过程中被误用的概率。
可能你又会说,可以给 AbstractBird 在派生出两个更加细分的抽象类:会飞的鸟 AbstractFlyableBird 和不会飞的鸟AbstractUnFlyableBird,让会飞的鸟继承 AbstractFlyableBird ,不会飞的鸟继承AbstractUnFlyableBird,这不就可以了吗?

上图中,可以看出,继承关系变成了三层。不过,整体上来讲,目前的继承关系还比较简单,层次比较浅。如果在加点难度,在刚刚的场景中,我们只关注鸟会不会非。但是,如果我们还关注“鸟会不会叫”,这个时候又该如何设计类之间的继承关系呢?
是否会飞?是否会叫?连个行为搭配起来会产生四种情况:
- 会飞会叫
- 会飞不会叫
- 不会飞会叫
- 不会飞不会叫
沿用上面的思路,那么还需要再定义四个抽象类:AbstractFlyableTweetableBird、AbstractFlyableUnTweetableBird、AbstractUnFlyableTweetableBird、AbstractUnFlyableUnTweetableBird。

如果还没还要关注:是否会下蛋,那估计组合就要爆炸了。类的继承层次会越来越深、继承关系会越来越复杂。这会导致两个方面的问题:
- 一方面,代码的可读性变差。我们要搞清楚类具有哪些属性和方法,必须阅读父类的代码、父类的父类的代码…,一直追溯到顶层父类的代码。
- 另一方面,也破坏了类的封装特性,将父类的实现细节暴露给了子类。子类的实现依赖父类的实现,两者高度耦合,一旦父类代码修改,就影响所有子类的逻辑。
所以,继承最大的问题在于:继承层次过深、继承关系过于复杂,会影响到代码的可读性和可维护性。这也是为什么不推荐使用继承的原因。
那这个问题又该如何解决呢?
组合比继承有哪些优势?
实际上,我们可以利用组合(composition)、接口、委托类(delegation)三个技术手段,一块来解决刚刚继承存在的问题。
接口表示具有某些特性的行为。针对会飞这样的特性,可以定义一个 Flyable 接口,只会让会飞的鸟去实现这个接口。对于会叫、会下蛋这些行为,我们可以类似的定义 Tweetable 接口和 EggLayable 接口。下面是具体的例子。
public interface Flyable {void fly();
}public interface Tweetable {void tween();
}public interface EggLayable{void layEgg();
}// 鸵鸟
public class Ostrich implements Tweetable, EggLayable {// 其他属性方法省略...@Overridepublic void tween() { /*...*/ }@Overridepublic void layEgg() { /*...*/ }
}// 麻雀
public class Sparrow implements Flyable, Tweetable, EggLayable {// 其他属性方法省略...@Overridepublic void fly() { /*...*/ }@Overridepublic void tween() { /*...*/ }@Overridepublic void layEgg() { /*...*/ }
}
只是,只生命接口,不定义实现的话,每个会下蛋的鸟都要实现一遍 layEgg() 方法,并且实现逻辑一样,这会导致代码重复。
我们可以针对这三个接口再定义三个实现类,分别是:
- 实现
Flyable的FlyAbility类 - 实现
Tweetable的TweetAbility类 - 实现
EggLayable的EggLayAbility类
然后通过组合和委托技术来消除代码重复。具体代码如下:
public interface Flyable {void fly();
}public class FlyAbility implements Flyable {// 其他属性方法省略...@Overridepublic void fly() { /*...*/ }
}
// 省略Tweetable/TweetAbility/EggLayable/EggLayAbilitypublic class Ostrich implements Tweetable, EggLayable {private TweetAbility tweetAbility = new TweetAbility();private EggLayAbility eggLayAbility = new EggLayAbility();// 其他属性方法省略...@Overridepublic void tween() {tweetAbility.tween(); // 委托}@Overridepublic void layEgg() {eggLayAbility.layEgg(); // 委托}
}
我们知道继承主要有三个作用:表示 is-a 关系,支持多态特性,代码复用。而这三个作用都可以通过其他技术手段来达成。
is-a关系:通过组合和接口的has-a关系来替代;- 多态特性:利用接口来实现
- 代码复用:通过组合和委托来实现。
所以,理论上,通过组合、接口、委托三个技术手段,完全可以替换掉继承,在项目中不用或者少用继承关系,特别是一些复杂的继承关系。
如何判断是该用组合还是继承?
尽管鼓励多用组合少用继承,但组合也并不是完美,继承也并非是一无是处。从上面的例子来看,继承改写成组合意味着要耕细粒度的类的拆分。这就以为这我们要定义更多的类和接口。类和接口的增多也就增加了代付复杂度和维护成本。所以,在实际的项目开发中,我们要根据情况,来具体选择用继承还是组合。
- 如果类之间的继承结构稳定(不会轻易改变),继承层次比较浅(比如,最多有两层继承关系),继承关系不复杂,我们就可以大胆地使用继承。
- 反之,系统越不稳定,继承层次很深,继承关系复杂,我们就尽量使用组合来替代继承。
初次之外,还有一些设计模式会固定使用继承或组合。比如,装饰者模式(decorator pattern)、策略模式(strategy pattern)、组合模式(composite pattern)等都使用了组合关系,而模板模式(template pattern)使用了继承关系。
前面我们讲过继承可以实现代码复用。利用继承特性,可以把相同的属性和方法,抽取出来,定义到父类中。子类复用父类的属性和方法,达到代码复用的目的。但是,有时候,从业务含义上,A 类和 B 类并一定具有继承关系。比如 Crawler 类和 PageAnalyzer 类,它们都用到了 URL 拼接和分割的功能,但并不具有继承关系(既不是父子,也不是兄弟)。仅仅为了代码复用,生硬地抽象出一个父类出来,会影响代码地可读性。这个时候,使用组合就更加合理、灵活。
public class Url {// 省略属性和方法...
}public class Crawler {private Url url;public Crawler() {this.url = new Url();}// ...
}public class PageAnalyzer {private Url url;public PageAnalyzer() {this.url = new Url();}// ...
}
如果有些场合要求必须使用继承。如果你不能改变一个函数的入参类型,而入参非接口,为了支持多态,只能采用继承来实现。比如下面的代码,其中 FeignClient 是一个外部类,我们没有权限去修改这部分代码,但是我们希望重写这个类在运行时执行的 encode() 函数。这个时候,我们只能采用继承实现了。
public class FeignClient {// 省略其他代码...public void encode(String url) { /*...*/ }
}public void demoFuction(FeignClient feignClient) {// ...feignClient.encode(url);// ...
}public class CustomizeFeignClient extend FeignClient {@Overridepublic void encode(String url) { /*重写encode的实现...*/ }
}// 调用
FeignClient client = new CustomizeFeignClient();
demoFuction(client);
尽管有些人,要杜绝继承,100% 用组合代替继承,但是我们可以不需要这么极端。之所以“多用组合少用继承”这个口号喊的响,只是因为,长期以来,我们过度使用继承。还是那句话,组合并不完美,继承也不是一无是处。只要我们控制好它的副作用、发挥它们各自的优势,在不同的场合下,恰当地选择使用组合还是继承,这才应该追求的境界。
总结
为什么不推荐使用继承?
继承是面向对象的四大特性之一,用来表示 is-a 的关系,可以解决代码复用的问题。虽然,继承的作用很多,但是继承层次太深、过复杂,也会影响到代码的可维护性。在这种情况下,应尽量少用,甚至不用继承。
组合相比继承有哪些优势?
继承主要有三个优势:表示 is-a 关系、支持多态、代码复用。而在三个作用可以通过组合、接口、委托三个技术手段来达成。初次之外,利用组合还能解决层次过深、过复杂的继承关系影响代码可维护性的问题。
如何判断该用组合还是继承?
在开发中需要根据情况来选择。如果类之间的继承结构稳定、层次比较浅、关系不复杂,那就可以大胆地使用继承。反之,就尽量使用组合来替代继承。初次之外,一些设计模式、特殊的场景,会固定使用继承或者组合。
相关文章:
设计模式学习笔记 - 面向对象 - 7.为什么要多用组合少用继承?如何决定该用组合还是继承?
前言 在面向对象编程中,有一条非常经典的设计原则:组合优于继承,多用组合少用继承。 为什么不推荐使用继承? 组合比继承有哪些优势? 如何判断该用组合还是继承? 为什么不推荐使用继承? 继承…...
RocketMQ生产环境常见问题分析与总结
RocketMQ生产环境常见问题分析与总结 如何保证消息不丢失 消息丢失场景 对于跨网络的节点可能会丢消息,因为MQ存盘都会先写入OS的PageCache中,然后再让OS进行异步刷盘,如果缓存中的数据未及时写入硬盘就会导致消息丢失 生产端到Broker端Brok…...
前端打包工具的发展历程、思路(grunt,gulp,webpack,vite)
现在前端发展真快,需要学的东西太多了,下面总结下前端打包的发展过程,便于区分和选择学习。 什么是前端打包 前端打包是指将多个JavaScript文件、CSS文件、图片等资源进行合并和优化处理,并输出为一个或多个文件的过程。这样做的目的是减少…...
利用Python将文件夹下多个txt文本写入到同一个excel中(每一个文件占一行)
1、 将文件夹下多个txt文本写入到同一个excel中(每一个文件占一行): # -*- coding: utf-8 -*- import os import pandas as pd# 获取文件夹中的所有txt文件 folder_path rG:\Cygwin\ txt_files [f for f in os.listdir(folder_path) if f.endswith(.t…...
通过Colab部署Google最新发布的Gemma模型
Gemma的简单介绍 Gemma 是一系列轻量级、最先进的开放式模型,采用与创建 Gemini 模型相同的研究和技术而构建。 Gemma 由 Google DeepMind 和 Google 的其他团队开发,其灵感来自 Gemini,其名称反映了拉丁语 gemma,意思是“宝石”…...
spring中@validate注解使用
在 Java 中,我们可以使用注解和 validate 实现对实体类中字段的校验。其中,注解用来定义字段的约束条件,而 validate 则用来进行实际的校验操作。 常用的校验注解包括 NotNull、NotEmpty、Size、Min、Max 等,它们可以帮助我们规定…...
停车场管理(C语言)
【题目描述】停车场管理。设有一个可以停放n辆汽车的狭长停车场,它只有一个大门可以供车辆进出。车辆按到达停车场时间的先后次序依次从停车场最里面向大门口处停放 (即最先到达的第一辆车停放在停车场的最里面) 。如果停车场已放满n辆车,则以后到达的车…...
探索无限:Sora与AI视频模型的技术革命 - 开创未来视觉艺术的新篇章
✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua,在这里我会分享我的知识和经验。&#x…...
375FPS! 谷歌提出MaskConver“重校正用于全景分割的纯卷积模型
https://arxiv.org/2312.06052 近年来,基于Transformer的模型由于其强大的建模能力以及对语义类和实例类的统一表示为全局二值掩码,在全景分割中占据主导地位。 在本文中,我们回顾了纯粹的卷积模型,并提出了一种新的结构MaskConve…...
leetcode初级算法(python)- 数组
文章目录 1.从排序数组中删除重复项2.买卖股票最佳时机23.旋转数组运行颠倒列表法整体移动元素块法4.存在重复运行包含判断法排序比较判断法运行集合判断法5.只出现一次的数字6.两个数组的交集27.移动零8.两数之和9.旋转图像这篇博客中的代码都是数组计算。 1.从排序数组中删除…...
重新定义音乐创作:ChatGPT与未来音乐产业的融合
### 重新定义音乐创作:ChatGPT与未来音乐产业的融合 随着人工智能技术的飞速发展,ChatGPT不仅在文字创作领域大放异彩,也正逐步渗透并重塑音乐产业的未来。这种先进的语言模型,如今已成为音乐家、作曲家和制作人们手中的一把利剑…...
人工智能绘画的时代下到底是谁在主导,是人类的想象力,还是AI的创造力?
#ai作画 目录 一.AI绘画的概念 1. 数据集准备: 2. 模型训练: 3. 生成绘画: 二.AI绘画的应用领域 三.AI绘画的发展 四.AI绘画背后的技术剖析 1.AI绘画的底层原理 2.主流模型的发展趋势 2.1VAE — 伊始之门 2.2GAN 2.2.1GAN相较于…...
[HTML]Web前端开发技术29(HTML5、CSS3、JavaScript )JavaScript基础——喵喵画网页
希望你开心,希望你健康,希望你幸福,希望你点赞! 最后的最后,关注喵,关注喵,关注喵,佬佬会看到更多有趣的博客哦!!! 喵喵喵,你对我真的很重要! 目录 前言 上一节的课后练习...
文本编辑器markdown语法
markdown语法 1.介绍 Markdown是一种使用一定的语法将普通的文本转换成HTML标签文本的编辑语言,它的特点是可以使用普通的文本编辑器来编写,只需要按照特定的语法标记就可以得到丰富多样的HTML格式的文本。 2.标题分级 "# " -> 一级标题 &…...
【C++】类和对象之拷贝构造函数篇
个人主页 : zxctscl 文章封面来自:艺术家–贤海林 如有转载请先通知 文章目录 1. 前言2. 传值传参和传引用传参3. 概念4. 特征 1. 前言 在前面学习了6个默认成员函数中的构造函数和析构函数 【C】构造函数和析构函数详解,接下来继续往后看拷…...
Mybatisplus 传参参数为自定义sql, 使用条件构造器作为参数
1 pom依赖 <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.5.3.1</version> </dependency> 2 mapper 接口文件 List<TBookOrder> searchDiy(Param(Const…...
C#与VisionPro联合开发——TCP/IP通信
TCP/IP(传输控制协议/互联网协议)是一组用于在网络上进行通信的通信协议。它是互联网和许多局域网的基础,为计算机之间的数据传输提供了可靠性、有序性和错误检测。在软件开发中,TCP/IP 通信通常用于实现网络应用程序之间的数据交…...
spring Boot快速入门
快速入门为主主要届介绍java web接口API的编写 java编辑器首选IntelliJ IDEA 官方链接:https://www.jetbrains.com/idea/ IEDA 前言 实例项目主要是web端API接口的使用,项目使用mysql数据库,把从数据库中的数据的查询出来后通过接口json数…...
FPGA SERDESE2 (SDR收发仿真)
高速 Serdes 环路测试 高速串行通信优势非常巨大,只需要很少的IO引脚就可以实现高速通信,这也是当今FPGA高速接口的核心 技术。比如XILINX的7代FPGA,GTX可以达到10.3125Gbps,ultrascale FPGA的GTH可以达到16Gbps。目前国产FPGA还难以达到这么高的接口速度。 高速串行通信经…...
Java异常体系结构核心解析-Throwable
资料不在于多,而在于精。好资料、好书,我们站在巨人的肩膀上前行,可以少走很多弯路。 通过搜索引擎找到自己需要的最好最权威信息,是一种很重要的能力。 Java源代码和官方资料Java™ Tutorials Java异常体系结构,是一种…...
Android Wi-Fi 连接失败日志分析
1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分: 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析: CTR…...
智慧医疗能源事业线深度画像分析(上)
引言 医疗行业作为现代社会的关键基础设施,其能源消耗与环境影响正日益受到关注。随着全球"双碳"目标的推进和可持续发展理念的深入,智慧医疗能源事业线应运而生,致力于通过创新技术与管理方案,重构医疗领域的能源使用模式。这一事业线融合了能源管理、可持续发…...
Leetcode 3577. Count the Number of Computer Unlocking Permutations
Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...
《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力
引言: 在人工智能快速发展的浪潮中,快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型(LLM)。该模型代表着该领域的重大突破,通过独特方式融合思考与非思考…...
【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:
根据万维钢精英日课6的内容,使用AI(2025)可以参考以下方法: 四个洞见 模型已经比人聪明:以ChatGPT o3为代表的AI非常强大,能运用高级理论解释道理、引用最新学术论文,生成对顶尖科学家都有用的…...
JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案
JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停 1. 安全点(Safepoint)阻塞 现象:JVM暂停但无GC日志,日志显示No GCs detected。原因:JVM等待所有线程进入安全点(如…...
在Spring Boot中集成RabbitMQ的完整指南
前言 在现代微服务架构中,消息队列(Message Queue)是实现异步通信、解耦系统组件的重要工具。RabbitMQ 是一个流行的消息中间件,支持多种消息协议,具有高可靠性和可扩展性。 本博客将详细介绍如何在 Spring Boot 项目…...
数据挖掘是什么?数据挖掘技术有哪些?
目录 一、数据挖掘是什么 二、常见的数据挖掘技术 1. 关联规则挖掘 2. 分类算法 3. 聚类分析 4. 回归分析 三、数据挖掘的应用领域 1. 商业领域 2. 医疗领域 3. 金融领域 4. 其他领域 四、数据挖掘面临的挑战和未来趋势 1. 面临的挑战 2. 未来趋势 五、总结 数据…...
