合成复用原则
合成复用原则 (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库,来实现一个省市区三级联动的组件。这个组件不仅可以提高用户体验,还能大大简化我们的代码。接下来…...
深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录
ASP.NET Core 是一个跨平台的开源框架,用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录,以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
vscode(仍待补充)
写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh? debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...
Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
04-初识css
一、css样式引入 1.1.内部样式 <div style"width: 100px;"></div>1.2.外部样式 1.2.1.外部样式1 <style>.aa {width: 100px;} </style> <div class"aa"></div>1.2.2.外部样式2 <!-- rel内表面引入的是style样…...
C++ 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...
dify打造数据可视化图表
一、概述 在日常工作和学习中,我们经常需要和数据打交道。无论是分析报告、项目展示,还是简单的数据洞察,一个清晰直观的图表,往往能胜过千言万语。 一款能让数据可视化变得超级简单的 MCP Server,由蚂蚁集团 AntV 团队…...
【Go语言基础【12】】指针:声明、取地址、解引用
文章目录 零、概述:指针 vs. 引用(类比其他语言)一、指针基础概念二、指针声明与初始化三、指针操作符1. &:取地址(拿到内存地址)2. *:解引用(拿到值) 四、空指针&am…...
WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...
