【Android】浅析六大设计原则

【Android】浅析六大设计原则

六大设计原则是软件开发中常用的设计原则,用来帮助开发者编写灵活、可维护、可扩展的代码。它们是面向对象设计(OOD)的核心,遵循这些原则能够避免代码中的常见问题,比如代码难以修改、难以扩展、难以维护等。

1. 单一职责原则(Single Responsibility Principle, SRP)

单一职责原则的核心思想是:一个类应该只有一个引起它变化的原因。换句话说,一个类应该只负责一项职责(即一个特定功能或行为),而不是承担多个职责。

一个类如果有多个职责,那么这些职责之间的耦合可能会让类变得难以维护。修改其中一个职责可能会影响另一个职责。单一职责原则的目的就是将不同的职责分离,使得每个类只承担一个职责,这样类的变化更为可控,代码更易于维护。

违反单一职责原则的常见问题:
  • 耦合过高:一个类承担多种职责时,这些职责会紧密耦合在一起。如果一个职责发生变化,可能会影响其他职责,导致系统中的多个模块被迫修改。
  • 难以维护:如果一个类有多个职责,随着时间的推移,代码会变得庞大且复杂,维护人员很难理解类的全部行为和目的。
  • 测试困难:单一职责原则下,每个类只负责一个功能,这样测试的复杂性较低。违反该原则的类则需要更复杂的测试,因为其内部包含多个不同的逻辑,需要更广泛的覆盖。
例子 1:员工类的职责划分

假设有一个Employee类,该类不仅负责员工的相关属性(如名字、薪水),还负责员工薪水计算、报告生成等其他职责。

class Employee {private String name;private double salary;public Employee(String name, double salary) {this.name = name;this.salary = salary;}public double calculateSalary() {// 计算薪水return salary;}public String generateReport() {// 生成员工报表return "Employee Report";}
}

在这个示例中,Employee类承担了两种职责:

  • 员工数据管理(如namesalary的管理)
  • 薪资计算和报表生成

这违反了单一职责原则,因为Employee类同时负责与员工数据和业务逻辑有关的内容。可以将薪资计算和报表生成职责从Employee类中分离出来。

改进后的设计:
class Employee {private String name;private double salary;public Employee(String name, double salary) {this.name = name;this.salary = salary;}public double getSalary() {return salary;}
}class SalaryCalculator {public double calculateSalary(Employee employee) {// 计算薪水return employee.getSalary();}
}class ReportGenerator {public String generateReport(Employee employee) {// 生成员工报表return "Report for " + employee;}
}

通过拆分Employee类,薪水计算和报表生成的职责被分别放在SalaryCalculatorReportGenerator类中。这样,如果将来薪资计算逻辑改变,或报表格式修改,我们只需要调整相应的类,而不必修改Employee类。

如何判断类是否违反了单一职责原则?
  • 类是否处理多种逻辑:如果一个类处理了多个领域的逻辑,比如业务逻辑、UI展示、数据存储等,可能违反了单一职责原则。
  • 类是否有多个原因需要修改:如果一个类因为多个不同的原因(如需求变化、逻辑修改)而需要频繁修改,说明它可能承担了过多的职责。
  • 类的方法是否过多:如果一个类的方法众多,而且方法之间的关联度不高,也可能是该类承担了多种职责的表现。
单一职责原则的扩展:

有时候,我们会对类的职责范围进行扩展,尤其是在组件化的架构中。可以引入多个层次的职责划分,例如:

  • 业务逻辑层:负责具体的业务操作和数据处理
  • 数据访问层:负责数据库或文件系统等数据存储操作
  • 服务层:封装业务逻辑和数据访问之间的交互

通过分层架构,单一职责原则可以被严格地应用到每个层次的类上。


2. 开放-关闭原则(Open-Closed Principle, OCP)

开放-关闭原则的定义是:软件实体(如类、模块、函数等)应该对扩展开放,对修改关闭。这意味着当需求变化时,软件系统应该允许通过扩展来引入新的功能,而不需要修改现有的代码。

开放-关闭原则的目的是减少代码修改的风险,并提高代码的稳定性和可维护性。遵循这个原则,可以在保持现有代码功能完整性的前提下,通过增加新的功能模块来扩展系统,而无需对已有代码进行修改。这种方式有助于避免引入新的 bug,并且使系统更加灵活。

开放-关闭原则的基本思路:
  • 对扩展开放:可以在不修改现有代码的情况下,新增新功能。例如,通过添加新类或方法来扩展系统的行为。
  • 对修改关闭:不应该修改已有的代码,尤其是核心业务逻辑的实现。修改现有代码可能会影响已经稳定运行的功能,增加出错的风险。
如何实现开放-关闭原则:

开放-关闭原则通常依赖于多态性继承接口编程,而非依赖具体实现。通过使用抽象类、接口和设计模式(如工厂模式、策略模式等),我们可以有效地在不修改原有代码的基础上进行扩展。

示例:

class Rectangle {public void draw() {// 画矩形}
}class Circle {public void draw() {// 画圆形}
}class ShapeManager {public void drawShapes(List<Object> shapes) {for (Object shape : shapes) {if (shape instanceof Rectangle) {((Rectangle) shape).draw();} else if (shape instanceof Circle) {((Circle) shape).draw();}}}
}

该示例中,ShapeManager需要修改以支持新的形状。遵循开放-关闭原则的解决方案是通过多态实现,例如创建一个Shape接口,所有形状类都实现它。

开放-关闭原则的实际应用:
  • 扩展现有功能:当系统需要引入新功能时,尽量避免直接修改现有的类或方法,而是通过继承、实现接口或添加新的模块来扩展现有功能。这保证了现有功能的稳定性和健壮性。
  • 避免代码重复:通过遵循开放-关闭原则,可以避免在多个地方重复修改同样的代码。修改某个模块时,只需要扩展特定模块,而不需要在系统的其他部分重复相同的逻辑。
  • 减少回归风险:当我们修改现有代码时,可能会无意间破坏已有的功能。而通过扩展功能而不是修改原有代码,可以减少这些回归问题。

3. 里氏替换原则(Liskov Substitution Principle, LSP)

定义: 子类对象应该能够替换父类对象,并且程序逻辑不受影响。

解释: 子类应当能够在父类能使用的任何地方替换父类,并且不改变程序的正确性。遵循里氏替换原则可以确保继承体系的健壮性,避免由于子类实现不当导致的逻辑错误。

示例:

class Bird {public void fly() {// 鸟会飞}
}class Ostrich extends Bird {@Overridepublic void fly() {throw new UnsupportedOperationException("鸵鸟不会飞");}
}

在这个例子中,Ostrich类违背了里氏替换原则,因为它不应该继承Bird类。如果代码中使用了Bird类型,却遇到Ostrich时会抛出异常,这显然是有问题的设计。


4. 依赖倒置原则(Dependency Inversion Principle, DIP)

定义: 高层模块不应该依赖于低层模块,二者都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。

解释: 通过依赖于抽象(接口或抽象类)而不是具体实现,可以使得高层模块不必关心低层模块的具体实现细节,从而减少耦合,提高模块的灵活性和可扩展性。

示例:

class LightBulb {public void turnOn() {// 打开灯泡}
}class Switch {private LightBulb lightBulb;public Switch(LightBulb lightBulb) {this.lightBulb = lightBulb;}public void operate() {lightBulb.turnOn();}
}

在此例中,Switch类依赖于具体的LightBulb类,违反了依赖倒置原则。我们可以通过引入抽象来改进:

interface Switchable {void turnOn();
}class LightBulb implements Switchable {public void turnOn() {// 打开灯泡}
}class Switch {private Switchable device;public Switch(Switchable device) {this.device = device;}public void operate() {device.turnOn();}
}

5. 接口隔离原则(Interface Segregation Principle, ISP)

定义: 不应该强迫用户依赖他们不需要的接口方法,接口应该尽量小而精。

解释: 大而全的接口会导致类实现一些无关的功能,而这些功能可能并不需要。通过将大接口拆分成多个小接口,类只需要实现其实际需要的功能接口,这样可以避免代码冗余,提高系统的灵活性和可维护性。

示例:

interface Worker {void work();void eat();
}

在这个示例中,Worker接口包含了workeat方法,可能对于某些实现类来说并不需要eat功能。可以将接口拆分为更小的接口:

interface Workable {void work();
}interface Eatable {void eat();
}

6. 迪米特法则(Law of Demeter, LoD)

迪米特法则,也称为最少知识原则(Principle of Least Knowledge),其核心思想是:一个对象应当尽可能少地了解其他对象的细节。具体来说,一个对象不应过多依赖于其他对象的内部结构和实现细节。对象之间的交互应该通过有限的、尽可能直接的接口进行,而不是通过一长串的调用链来操作多个对象的内部数据。

迪米特法则鼓励我们设计低耦合的系统。高耦合的系统会导致修改某个模块时不得不改动许多依赖模块,增加了系统复杂性和维护难度。通过限制对象之间的过度依赖,迪米特法则减少了耦合,提高了系统的灵活性和可维护性。

具体规则:

根据迪米特法则,一个对象只应与以下几种“朋友”通信:

  1. 自身:对象可以与自身的方法和属性交互。
  2. 成员对象:对象可以与直接包含的成员对象交互。
  3. 方法参数:对象可以与方法参数交互。
  4. 创建的对象:对象可以与它自己创建的对象交互。
  5. 全局变量或单例对象:在某些设计中,对象可以与全局对象或单例对象交互(但应当谨慎使用)。

示例:

class Customer {public Wallet getWallet() {return new Wallet();}
}class Wallet {public void deductAmount(int amount) {// 扣款}
}class Store {public void processPayment(Customer customer, int amount) {customer.getWallet().deductAmount(amount);}
}

这里Store类直接依赖于CustomerWallet类,违反了迪米特法则。更好的做法是通过Customer提供一个支付方法,而不暴露内部的Wallet

class Customer {private Wallet wallet;public void pay(int amount) {wallet.deductAmount(amount);}
}class Store {public void processPayment(Customer customer, int amount) {customer.pay(amount);}
}

结语

参考:

设计模式之六大设计原则-CSDN博客

【设计模式】六大设计原则_设计模式6大准则-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/55343.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue页面,基础配置

最简单页面 日期范围及字符搜索&#xff0c;监听器处理日期范围搜索控件清空重置问题导出、导出文件文件名称带日期时间表格日期指定格式显示。。。 <template><div class"app-container"><el-form :model"queryParams" ref"queryForm…

YOLOv11改进策略【损失函数篇】| Shape-IoU:考虑边界框形状和尺度的更精确度量

一、本文介绍 本文记录的是改进YOLOv11的损失函数&#xff0c;将其替换成Shape-IoU。现有边界框回归方法通常考虑真实GT&#xff08;Ground Truth&#xff09;框与预测框之间的几何关系&#xff0c;通过边界框的相对位置和形状计算损失&#xff0c;但忽略了边界框本身的形状和…

uboot笔记(一):概括/移植等

目录 前言一、下载地址二、目录介绍三、编译四、移植/适配五、启动流程 前言 本笔记以u-boot-2024.10-rc4代码、在arm64平台运行为例对uboot的介绍、编译、适配移植、运行过程的关键流程等&#xff1b; 一、下载地址 https://ftp.denx.de/pub/u-boot/ 下载自己想要使用的版本即…

并发编程三大特性(原子性、可见性、有序性)

并发编程的三大特性实际是JVM规范要求的JVM实现必须保证的三大特性 不同的硬件和不同的操作系统在内存管理上有一定的差异&#xff0c;JAVA为了解决这种差异&#xff0c;使用JMM&#xff08;Java Memry Model&#xff09;来屏蔽各个操作系统之间的差异&#xff0c;使得java可以…

关于malloc,calloc,realloc

1.引用的头文件介绍&#xff1a; 这三个函数需要调用<stdlib.h>这个头文件 2.malloc 2.1 函数简单介绍&#xff1a; 首先这个函数是用于动态开辟一个空间&#xff0c;例如数组在c99标准之前是无法arr[N]的&#xff0c;这个时候就需要使用malloc去进行处理&#xff0c…

互斥量mutex、锁、条件变量和信号量相关原语(函数)----很全

线程相关知识可以看这里: 线程控制原语(函数)的介绍-CSDN博客 进程组、会话、守护进程和线程的概念-CSDN博客 1.同步概念 所谓同步&#xff0c;即同时起步&#xff0c;协调一致。不同的对象&#xff0c;对“同步”的理解方式略有不同。如&#xff0c;设备同步&#xff0c;是…

【C语言指南】数据类型详解(上)——内置类型

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;《C语言指南》 期待您的关注 目录 引言 1. 整型&#xff08;Integer Types&#xff09; 2. 浮点型&#xff08;Floating-Point …

计算机毕业设计 基于Python高校岗位招聘和分析平台的设计与实现 Python+Django+Vue 前后端分离 附源码 讲解 文档

&#x1f34a;作者&#xff1a;计算机编程-吉哥 &#x1f34a;简介&#xff1a;专业从事JavaWeb程序开发&#xff0c;微信小程序开发&#xff0c;定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事&#xff0c;生活就是快乐的。 &#x1f34a;心愿&#xff1a;点…

Ruby基础语法

Ruby 是一种动态、反射和面向对象的编程语言&#xff0c;它以其简洁的语法和强大的功能而受到许多开发者的喜爱。以下是 Ruby 语言的一些基本语法&#xff1a; 1. 打印输出 puts "Hello, Ruby!" 变量赋值 x 10 name "John" 2. 数据类型 Ruby 有多种…

YOLOv8改进 ,YOLOv8改进主干网络为华为的轻量化架构GhostNetV1

摘要 摘要:将卷积神经网络(CNN)部署在嵌入式设备上是困难的,因为嵌入式设备的内存和计算资源有限。特征图的冗余是成功的 CNN 的一个重要特征,但在神经网络架构设计中很少被研究。作者提出了一种新颖的 Ghost 模块,用于通过廉价操作生成更多的特征图。基于一组内在特征图…

力扣(leetcode)每日一题 983 最低票价 |动态规划

983. 最低票价 题干 在一个火车旅行很受欢迎的国度&#xff0c;你提前一年计划了一些火车旅行。在接下来的一年里&#xff0c;你要旅行的日子将以一个名为 days 的数组给出。每一项是一个从 1 到 365 的整数。 火车票有 三种不同的销售方式 &#xff1a; 一张 为期一天 的通…

Android 安卓内存安全漏洞数量大幅下降的原因

谷歌决定使用内存安全的编程语言 Rust 向 Android 代码库中写入新代码&#xff0c;尽管旧代码&#xff08;用 C/C 编写&#xff09;没有被重写&#xff0c;但内存安全漏洞却大幅减少。 Android 代码库中每年发现的内存安全漏洞数量&#xff08;来源&#xff1a;谷歌&#xff09…

Spring Boot实现足球青训俱乐部管理自动化

4 系统设计 4.1 系统架构设计 B/S系统架构是本系统开发采用的结构模式&#xff0c;使用B/S模式开发程序以及程序后期维护层面需要的经济成本是很低的&#xff0c;用户能够承担得起。使用这样的模式开发&#xff0c;用户使用起来舒心愉悦&#xff0c;不会觉得别扭&#xff0c;操…

WebSocket消息防丢ACK和心跳机制对信息安全性的作用及实现方法

WebSocket消息防丢ACK和心跳机制对信息安全性的作用及实现方法 在现代即时通讯&#xff08;IM&#xff09;系统和实时通信应用中&#xff0c;WebSocket作为一种高效的双向通信协议&#xff0c;得到了广泛应用。然而&#xff0c;在实际使用中&#xff0c;如何确保消息的可靠传输…

Docker笔记-Docker磁盘空间清理

无用的容器指的是已经停止运行且处于非活跃状态的容器。无用的镜像包括没有被任何容器使用的镜像&#xff0c;或者是被标记为"<none>"的镜像&#xff0c;通常是构建过程中产生的无标签镜像。 通过执行 docker container ls -a 和 docker image ls -a 命令&…

ffmpeg录制视频功能

本文目录 1.环境配置2.ffmpeg编解码的主要逻辑&#xff1a;3. 捕获屏幕帧与写入输出文件4. 释放资源 在录制结束时&#xff0c;释放所有分配的资源。5.自定义I/O上下文6.对于ACC编码器注意事项 1.环境配置 下载并安装FFmpeg库 在Windows上 从FFmpeg官方网站下载预编译的FFmpeg…

LiveNVR监控流媒体Onvif/RTSP功能-支持电子放大拉框放大直播视频拉框放大录像视频流拉框放大电子放大

LiveNVR监控流媒体Onvif/RTSP功能-支持电子放大拉框放大直播视频拉框放大录像视频流拉框放大电子放大 1、视频广场2、录像回看3、RTSP/HLS/FLV/RTMP拉流Onvif流媒体服务 1、视频广场 视频广场 -》播放 &#xff0c;左键单击可以拉取矩形框&#xff0c;放大选中的范围&#xff…

使用conda-pack迁移环境

要使用 conda-pack 迁移 conda 环境&#xff0c;可以按照以下步骤进行&#xff1a; 安装 conda-pack: 首先&#xff0c;需要确保 conda-pack 已安装。如果没有安装&#xff0c;可以通过 pip 安装&#xff1a; pip install conda-pack打包环境: 选择你想要打包的 conda 环境&…

2024年9月中国干旱监测报告(FYDI-2.0指数)

目录 引言 旱情监测与分析 资料来源 引言 2024年9月&#xff0c;北方的降水逐渐增多&#xff0c;进入华西秋雨集中期&#xff0c;从青藏高原北部一直延伸到东北多地&#xff0c;常出现大范围的云带&#xff0c;西北地区的降雨强度较大。南方地区降水分布不均&#xff0c;受…

【Python报错已解决】error: subprocess-exited-with-error

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 专栏介绍 在软件开发和日常使用中&#xff0c;BUG是不可避免的。本专栏致力于为广大开发者和技术爱好者提供一个关于BUG解决的经…