合成复用原则
合成复用原则 (Composite Reuse Principle, CRP)
合成复用原则(Composite Reuse Principle, CRP),也被称为组合/聚合复用原则,是面向对象设计中的一条重要原则。它的核心思想是:优先使用对象组合/聚合,而不是通过继承来实现代码复用。该原则旨在提高代码的灵活性和可维护性,减少类之间的紧密耦合。
1. 原则解释
合成复用原则强调,通过组合或聚合多个对象来实现新的功能,而不是通过继承来扩展类的功能。它建议在设计系统时,尽量使用组合或聚合来构建复杂对象,只有在明确需要继承的情况下才使用继承。
- 继承:表示类与类之间的“是一个”的关系(is-a)。子类继承父类的所有特性和行为,是一种强耦合关系。
- 组合:表示类与类之间的“有一个”的关系(has-a)。一个类通过包含其他类的实例来实现功能,是一种松耦合关系。
- 聚合:表示类与类之间的“整体-部分”的关系(whole-part)。类似于组合,但更加松散。
2. 违反合成复用原则的例子
假设我们有一个简单的动物系统,包括鸟类和鱼类。我们可能会首先设计一个基类 Animal,并通过继承来扩展不同的动物类:
public class Animal {public void move() {System.out.println("Animal moves");}
}public class Bird extends Animal {@Overridepublic void move() {System.out.println("Bird flies");}
}public class Fish extends Animal {@Overridepublic void move() {System.out.println("Fish swims");}
}
在这个设计中,Bird 和 Fish 类继承了 Animal 类,并分别重写了 move 方法。然而,如果我们需要进一步扩展,例如添加更多的行为(如吃饭、睡觉),这种设计会变得复杂且难以维护。
3. 遵循合成复用原则的改进
为了遵循合成复用原则,我们可以通过组合的方式来实现新的功能,而不是通过继承。下面是一个改进的设计:
// 移动行为接口
public interface MoveBehavior {void move();
}// 飞行行为
public class FlyBehavior implements MoveBehavior {@Overridepublic void move() {System.out.println("Bird flies");}
}// 游泳行为
public class SwimBehavior implements MoveBehavior {@Overridepublic void move() {System.out.println("Fish swims");}
}// 动物类
public class Animal {private MoveBehavior moveBehavior;public Animal(MoveBehavior moveBehavior) {this.moveBehavior = moveBehavior;}public void performMove() {moveBehavior.move();}public void setMoveBehavior(MoveBehavior moveBehavior) {this.moveBehavior = moveBehavior;}
}// 主类
public class Main {public static void main(String[] args) {Animal bird = new Animal(new FlyBehavior());bird.performMove();Animal fish = new Animal(new SwimBehavior());fish.performMove();// 动态改变行为bird.setMoveBehavior(new SwimBehavior());bird.performMove();}
}
在这个改进后的设计中,我们定义了一个 MoveBehavior 接口,并实现了具体的移动行为(FlyBehavior 和 SwimBehavior)。Animal 类通过组合 MoveBehavior 来实现不同的移动行为。这样,我们可以在运行时动态地改变动物的行为,而不需要修改或继承类。
4. 具体使用示例
让我们来看一个更复杂的例子,展示如何在实际开发中遵循合成复用原则。
// 支付方式接口
public interface PaymentMethod {void pay(double amount);
}// 信用卡支付
public class CreditCardPayment implements PaymentMethod {@Overridepublic void pay(double amount) {System.out.println("Paid " + amount + " using Credit Card");}
}// 支付宝支付
public class AlipayPayment implements PaymentMethod {@Overridepublic void pay(double amount) {System.out.println("Paid " + amount + " using Alipay");}
}// 用户类
public class User {private PaymentMethod paymentMethod;public User(PaymentMethod paymentMethod) {this.paymentMethod = paymentMethod;}public void makePayment(double amount) {paymentMethod.pay(amount);}public void setPaymentMethod(PaymentMethod paymentMethod) {this.paymentMethod = paymentMethod;}
}// 主类
public class Main {public static void main(String[] args) {User user = new User(new CreditCardPayment());user.makePayment(100.0);// 动态改变支付方式user.setPaymentMethod(new AlipayPayment());user.makePayment(200.0);}
}
在这个例子中,我们定义了一个 PaymentMethod 接口,并实现了不同的支付方式(CreditCardPayment 和 AlipayPayment)。User 类通过组合 PaymentMethod 来实现支付功能。这样,我们可以在运行时动态地改变用户的支付方式,而不需要修改或继承类。
5. 总结
合成复用原则是面向对象设计中的基本原则之一,通过优先使用组合或聚合而不是继承,可以提高系统的灵活性和可维护性。在实际开发中,遵循合成复用原则有助于我们设计出高质量的代码,使系统更加稳定和易于扩展。
希望这个博客对你有所帮助。如果你有任何问题或需要进一步的例子,请随时告诉我!
相关文章:
合成复用原则
合成复用原则 (Composite Reuse Principle, CRP) 合成复用原则(Composite Reuse Principle, CRP),也被称为组合/聚合复用原则,是面向对象设计中的一条重要原则。它的核心思想是:优先使用对象组合/聚合,而不…...
安卓自带camera hal3 实例README.md翻译
最近,遇到一个这样的问题,临时了解下这个驱动实现架构和特点,翻译如下 V4L2相机HALv3 camera.v4l2库使用视频Linux 2(V4L2)接口实现了camera HAL v3。这使得它在理论上可以与各种设备配合使用,尽管V4L2的…...
ActiViz实战:ActiViz中的自己实现鼠标双击事件
文章目录 1、添加鼠标事件2、网上实现双击事件的方式3、增加双击的时间限制4、补充说明1、添加鼠标事件 已知在C#中观察者/命令模式会报错,正常添加鼠标事件如下: private void VtkInteractorStyleTest() {vtkInteractorStyle style = vtkInteractorStyle.New();style.LeftB…...
Linux-交换空间(Swap)管理
引入概念 在计算机中,硬盘的容量一般比内存大,内存(4GB 8GB 16GB 32GB 64GB…),硬盘(512GB 1T 2T…)。 冯诺依曼的现代计算机结构体系里面的存储器就是内存 内存是一种易失性存储器,…...
扫描某个网段下存活的IP:fping
前言: 之前用arp统计过某网段下的ip,但是有可能统计不全。网络管理平台又不允许登录。想要知道当前的ip占用情况,可以使用fping fping命令类似于ping,但比ping更强大。与ping需要等待某一主机连接超时或发回反馈信息不同&#x…...
【深度学习】PyTorch框架(3):优化与初始化
1.引言 在本文中,我们将探讨神经网络的优化与初始化技术。随着神经网络深度的增加,我们会遇到多种挑战。最关键的是确保网络中梯度流动的稳定性,否则可能会遭遇梯度消失或梯度爆炸的问题。因此,我们将深入探讨以下两个核心概念&a…...
Go-知识测试-子测试
Go-知识测试-子测试 1. 介绍2. 例子3. 子测试命名规则4. 选择性执行5. 子测试并发6. testing.T.Run7. testing.T.Parallel8. 子测试适用于单元测试9. 子测试适用于性能测试10. 总结10.1 启动子测试 Run10.2 启动并发测试 Parallel 建议先看:https://blog.csdn.net/a…...
.net core IConfiguration 读 appsettings.json 数据,举例
在.NET Core中,IConfiguration 接口是用来读取配置数据的,包括从 appsettings.json 文件中读取。下面是一个如何在使用.NET Core时通过 IConfiguration 读取 appsettings.json 数据的示例。 首先,假设你的 appsettings.json 文件内容如下&am…...
全球Windows机器蓝屏,作为量化人,我的检讨来了
昨天下午,微软给大家放了个假。Windows又双叒死机了。不过,这一次不是几台机器,而是全球大范围宕机。这一刻,大家都是“正蓝旗”。 蓝瓶的,效果好! 现在根本原因已经找到,绝大多数人的机器都已修…...
部署和运维
目录 1.Git1.1. Git指令中merge和rebase的区别1. Commit 记录2. 合并方式3. 冲突处理4. 使用场景选择建议 1.2. cherry-pick的使用如何使用 git cherry-pick例子处理冲突撤销 cherry-pick其他选项 结论 2. 部署1. Nginx的使用场景 编译打包1. webpack2. webpack打包优化1. 代码…...
微信小程序基本语法
官网 https://developers.weixin.qq.com/miniprogram/dev/framework/ 视频教程:尚硅谷微信小程序开发教程,2024最新微信小程序项目实战! 仿慕尚花坊项目源码:https://gitee.com/abcdfdewrw/flower-workshop 目录 一,初…...
测试用例的设计方法
等价类 等价类概念:在所有测试的数据中,具有某种共同特征的数据子集 边界值 边界值分析是对程序输入或输出的边界值进行测试的一种黑盒测试方法 边界值是作为等价类的补充,其主要区别是: 边界值测试设计不是从某一个等价类中…...
Android10.0 锁屏分析-KeyguardPatternView图案锁分析
首先一起看看下面这张图: 通过前面锁屏加载流程可以知道在KeyguardSecurityContainer中使用getSecurityView()根据不同的securityMode inflate出来,并添加到界面上的。 我们知道,Pattern锁所使用的layout是 R.layout.keyguard_pattern_view&a…...
Python 装饰器:函数的函数,代码的艺术
引言 在Python中,装饰器是一种强大的功能,允许程序员在不修改原函数源码的情况下增强或修改函数行为。装饰器本质上是一个接收函数作为参数的高阶函数,并返回一个新的函数或修改原函数的行为。这种机制极大地提高了代码的复用性、可读性和模…...
安全防御2
实验要求: 实验过程: 7,办公区设备可以通过电信链路和移动链路上网(多对多的NAT,并且需要保留一个公网IP不能用来转换): 新建电信区: 新建移动区: 将对应接口划归到各自区域: 新建…...
C语言 ——— 打印水仙花数
目录 何为水仙花数 题目要求 代码实现 何为水仙花数 “水仙花数”是指一个n位数,其各位数字的n次方之和等于该数本身 如:153 1^3 5^3 3^3,则153就是一个“水仙花数” 题目要求 求出0~100000的所有“水仙花数”并输出 代码实现 #i…...
「Conda」在Linux系统中安装Conda环境管理器
在Linux系统中安装Conda环境管理器是一个相对简单的过程。 1. 准备工作 确保你的Linux系统已经更新到最新版本,并安装了基本的开发工具和库。打开终端,执行以下命令: sudo apt-get update sudo apt-get upgrade sudo apt-get install build-essential2. 安装Miniconda或An…...
9.11和9.9哪个大?GPT-4o也翻车了
今天刷到了这个问题,心血来潮去问下chatgpt-4o,没想到疯狂翻车... 第一次问: GPT一开始给出了难绷的解答,让我想起了某短视频软件评论区里对某歌手节目排名的质疑哈哈哈哈哈 但是在接下来的进一步询问和回答中它反应过来了。 第…...
[开源]语雀+Vercel:打造免费个人博客网站
大家好,我是白露。 今天我想和大家分享我的今年的第一个开源项目 —— 基于语雀+Nextjs+Vercel实现免费的博客系统。 简单来说,你在语雀写博客,然后直接一键同步到个人网站上,网站自动部署! 而且,整个过程几乎不需要额外的成本,也不用充值语雀超级会员,hh。这个项目…...
使用ElementUI和element-china-area-data库实现省市区三级联动组件封装
在前端开发中,省市区三级联动是一个常见的需求。今天我们将使用Vue.js和ElementUI组件库,结合element-china-area-data库,来实现一个省市区三级联动的组件。这个组件不仅可以提高用户体验,还能大大简化我们的代码。接下来…...
镜头结构设计中的公差与成本平衡:如何避免过度设计
镜头结构设计中的公差与成本平衡:如何避免过度设计 在高端光学镜头的研发过程中,工程师们常常面临一个核心矛盾:如何在确保光学性能的同时,避免因过度追求精度而导致生产成本失控?这个看似简单的平衡问题,实…...
Pixel Mind Decoder 嵌入式应用初探:STM32设备日志情绪分析
Pixel Mind Decoder 嵌入式应用初探:STM32设备日志情绪分析 1. 场景痛点与解决方案 在工业物联网领域,设备维护一直是个让人头疼的问题。想象一下,工厂里几十台STM32设备日夜运转,工程师们每天要盯着密密麻麻的日志数据…...
Vue Toast组件:轻量级通知解决方案的无侵入式集成实践
Vue Toast组件:轻量级通知解决方案的无侵入式集成实践 【免费下载链接】vue-sonner 🔔 An opinionated toast component for Vue. 项目地址: https://gitcode.com/gh_mirrors/vu/vue-sonner 在现代Web应用开发中,用户交互反馈是提升体…...
移动开发环境解决方案:VSCode Portable技术解析与实践指南
移动开发环境解决方案:VSCode Portable技术解析与实践指南 【免费下载链接】VSCode-Portable VSCode 便携版 VSCode Portable 项目地址: https://gitcode.com/gh_mirrors/vsc/VSCode-Portable 在现代软件开发流程中,开发环境的一致性与可迁移性已…...
java Ai开发工具①
笼统了解大模型是个啥?大模型就像个“全能型天才”,天生啥都懂点,你再教它一门手艺(比如Java),它就能快速上手并成为高手——但它骨子里还是那个会聊天、会思考、会举一反三的“通才”。三种工具ǹ…...
Qwen3.5-4B-Claude-Opus基础教程:Q4_K_M量化精度与响应速度平衡
Qwen3.5-4B-Claude-Opus基础教程:Q4_K_M量化精度与响应速度平衡 1. 模型概述 Qwen3.5-4B-Claude-4.6-Opus-Reasoning-Distilled-GGUF是一个基于Qwen3.5-4B架构的推理蒸馏模型,特别强化了结构化分析、分步骤回答以及代码与逻辑类问题的处理能力。该模型…...
六、免Root免Hook,探索Android原生系统层定位修改新实践
1. 为什么需要免Root免Hook的定位修改方案 最近几年,金融类App的风控系统越来越严格,传统的定位修改方法逐渐失效。我做过测试,某银行App能检测出90%的Xposed框架,甚至能识别Magisk的隐藏模式。更麻烦的是,很多社交平…...
伏羲天气预报开放科学:复现代码、数据、环境全公开,推动可重复研究
伏羲天气预报开放科学:复现代码、数据、环境全公开,推动可重复研究 天气预报,这个与每个人生活息息相关的领域,正经历着一场由人工智能驱动的深刻变革。传统的数值天气预报模型虽然精度高,但计算成本巨大,…...
核心理念:Spec Kit如何重塑AI驱动的软件开发流程
核心理念:Spec Kit如何重塑AI驱动的软件开发流程 【免费下载链接】spec-kit 💫 Toolkit to help you get started with Spec-Driven Development 项目地址: https://gitcode.com/gh_mirrors/sp/spec-kit 在传统软件开发中,你是否经常遇…...
预算有限必入!4 款高性价比买断制写作软件,学生党低成本搞定全流程写作
在写作成本日益攀升的当下,买断制写作软件凭借 “一次付费、长期使用、无隐性消费” 的优势,成为学生党、自由创作者与职场人的首选。尤其对于预算有限的群体,既能规避订阅制的长期支出,又能解锁全流程写作能力,堪称 “…...
