【设计模式】第2节:七大设计原则

news2025/1/11 2:39:50

一、七大设计原则

七大原则提出的目的是降低对象之间的耦合度,提高程序的可复用性、可扩展性和可维护性。

请添加图片描述

1.单一职责原则

Single Responsibility Principle,SRP原则:一个类只负责一个功能领域中的相应职责。或者可以定义为:就一个类而言,应该只有一个引起它变化的原因。单一职责原则是实现高内聚、低耦合的指导方针。

  • 降低类的负责度,一个类只负责一项职责。
  • 提高类的可读性,可维护性。
  • 降低变更带来的风险。

2. 开放封闭原则

Open Closed Principle,OCP原则:需求改变时,在不改变软件实体源代码(类、接口、方法等)的前提下,通过扩展功能,使其满足新的需求。功能是开放的(功能提供方),代码修改是封闭的(功能使用方)。

  • 用抽象构建框架,用实现扩展细节
  • 参数类型、引用对象尽量使用接口或抽象类
  • 抽象层尽量保持稳定:接口和抽象类只负责定义方法,但不负责具体实现

3. 里式替换原则

Liskov Substitution Principle,LSP原则:所有引用基类(父类)的地方必须能透明地使用其子类的对象。里氏代换原则表明,在软件中将一个基类对象替换成它的子类对象,程序将不会产生任何错误和异常,反过来则不成立。该原则实际是在使用继承关系时的指导原则,遵循了该原则,那么采用继承时就不容易出错。

  • 子类必须完全实现父类的抽象方法,但不能覆盖父类的非抽象方法
  • 子类可以实现自己特有的方法
  • 当子类的方法实现父类的抽象方法时,方法的后置条件(即方法的返回值)要比父类更严格‘
  • 子类的实例可以替代任何父类的实例,但反之不成立

4.接口隔离原则

Interface Segragation Principle,ISP原则:客户端不应该依赖它不需要的接口,即一个类对另一个类的依赖应该建立在最小的接口上。接口尽量细分,不要在一个接口中放很多方法。

接口分离原则和单一职责原则的关系:单一职责原则是为了高内聚,接口分离原则是为了低耦合。

5.依赖倒置原则

Dependence Inversion Principle,DIP原则:

  • 高层模块不应该依赖底层模块,二者都应该依赖其抽象(接口)。
  • 抽象不应该依赖细节,细节应该依赖抽象。

请添加图片描述

依赖倒置的本质是通过抽象(接口或抽象类)是各个类或模块的实现彼此独立,互不影响,实现模块间的低耦合。相对于细节的多变性,抽象的东西要稳定的多。应用依赖倒置的设计模式:工厂方法。

class BMW {
  public void rentBMW(String model) {
    System.out.println("BMW rented" + model);
  }
}

class Mercedes {
  public void rentMercedes(String model) {
    System.out.println("Mercedes rented" + model);
  }
}

public class CarRentalAgency {
  public static void main(String[] args) {
    CarRentalAgency agency = new CarRentalAgency();
    agency.rentCar("BMW", "X5");
    agency.rentCar("Mercedes", "GLE");
  }
  
  public void rentCar(String brand, String model) {
    if (brand == "BMW") {
      new BMW().rentBMW(model);      
    } else if (brand == "Mercedes") {
      new Mercedes().rentMercedes(model);      
    }
  }
}
interface CarManufactory {
  public void rent(String model);
}

class BMW implements CarManufactory {
  public void rent(String model) {
    System.out.println("BMW rented" + model);
  }
}

class Mercedes implements CarManufactory {
  public void rent(String model) {
    System.out.println("Mercedes rented" + model);
  }
}

class Honda implements CarManufactory {
  public void rent(String model) {
    System.out.println("Honda rented" + model);
  }
}

public class CarRentalAgency {
  public static void main(String[] args) {
    CarRentalAgency agency = new CarRentalAgency();
    agency.RentCar(new BMW(), "X5");
    agency.RentCar(new Mercedes(), "GLE");
    agency.RentCar(new Honda(), "Accord");
  }
  
  pubcic void rentCar(CarManufactory cm, String mode) {
    cm.rent(mode);
  }
}

一个优秀的面向对象的程序设计,核心原则之一就是将变化隔离/封装,使得变化的部分发生变化时,对其他部分无影响。为了实现这个目的,需要使用面向接口编程,使用后,客户类不再直接依赖服务类,而是依赖一个抽象的接口,这样,客户类就不能在内部直接实例化服务类。但是客户类在运行的过程中,又需要具体的服务类来提供服务,因为接口是不能实例化的,就产生了一个矛盾:客户类不允许实例化服务类,但是客户类又需要服务类的服务。为了解决这个矛盾,设计了一种解决方案:客户类定义一个注入点,用于服务类的注入,而客户类负责根据情况,实例化服务类,注入到客户类中,从而解决了这个矛盾,该过程即依赖注入。

依赖注入的三种方式:

  • 接口注入:在接口/类中,将要注入的服务对象,以参数的形式直接注入
  • 构造方法注入:通过构造函数来传入具体类的对象
  • set方法注入:通过Setter方法来传入具体类的对象

开闭原则是目标,里氏代换原则是基础,依赖倒转原则是手段,它们相互补充,相辅相成,目标一致,只是分析问题时所站角度不同而已。以上五个原则根据其首字母,又被称为SOLID原则。

6.合成复用原则

尽量使用对象聚合/ 组合,而不是继承来达到复用的目的。

请添加图片描述

继承复用:

  • 优点:简单,容易实现
  • 缺点:破坏了封装性,耦合度高,限制灵活性

合成复用:

  • 优点:维持封装性,降低耦合度,灵活性高
  • 缺点:有较多的对象需要管理

7.迪米特原则

The Least Knowledge Principe,/Demeter Principle:它要求一个对象应该对其他对象有最少的了解(最少知识原则),降低类之间的耦合。迪米特原则实际上就是一个类在创建方法和属性时要遵守的法则。

迪米特法则还有一种定义形式:不要和“陌生人”说话,只与你的直接朋友通信等。在迪米特法则中,对于一个对象,其“朋友”包括以下几类:

  • 当前对象本身(this);
  • 以参数形式传入到当前对象方法中的对象;
  • 当前对象的成员对象;
  • 如果当前对象的成员对象是一个集合,那么集合中的元素也都是朋友;
  • 当前对象所创建的对象。

遵循迪米特原则的设计模式:

  • 门面模式
  • 中介模式
class Card {
  public int balance = 10;
}

class Customer {
  private Card card = new Card();
  
  public Card getCard() {
    return this.card;
  }
}

public class SurfShop {
  public static void main(String[] args) {
    SurfShop surfShop = new SurfShop();
    Customer customer = new Customer();
    surfShop.chargeCustomer(customer, 10);
    System.out.println("Done");
  }
  
  public void chargeCustomer(Customer c, float fee) {
    c.getCard().balance -= fee;
  }
}
class Card {
  public int balance = 10;
  
  public void deduct(flaot fee) {
    this.balance -= fee;
  }
}

class Customer {
  private Card card = new Card();
  
  public void pay(float fee) {
		this.card.deduct(fee);
  }
}

public class SurfShop {
  public static void main(String[] args) {
    SurfShop surfShop = new SurfShop();
    Customer customer = new Customer();
    surfShop.chargeCustomer(customer, 10);
    System.out.println("Done");
  }
  
  public void chargeCustomer(Customer c, float fee) {
    c.pay(fee);
  }
}

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

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

相关文章

安防视频监控平台EasyNVR平台如何在角色表格中添加信息

TSINGSEE青犀视频安防监控平台EasyNVR可支持设备通过RTSP/Onvif协议接入,并能对接入的视频流进行处理与多端分发,包括RTSP、RTMP、HTTP-FLV、WS-FLV、HLS、WebRTC等多种格式。在智慧安防等视频监控场景中,EasyNVR可提供视频实时监控直播、云端…

Linux | 进程

目录 前言 一、什么是进程 1、Window下的进程 2、深刻理解进程 3、见一见Linux下的进程 二、进程的创建 1、初始fork 2、fork的返回值 3、深刻理解fork函数 4、遗留问题 三、进程状态 1、操作系统中的进程状态 2、Linux中的进程状态 3、状态演示 &#xff08…

MySQL 概述 数据库表操作 数据增删改

目录 MySQL概述前言安装与配置MySQL登录与卸载 数据模型概述SQL简介SQL通用语法简介SQL分类 数据库设计(数据库操作)-DDL数据库操作查询数据库创建数据库使用数据库删除数据库 图形化工具连接数据库操作数据库 表操作创建表创建表语法创建表约束语法创建表数据类型 查询表修改表…

【Python3】【力扣题】169. 多数元素

【力扣题】题目描述: 众数:一组数据中出现次数最多的数据。 【Python3】代码: 1、解题思路:哈希表。使用哈希映射存储各元素以及出现的次数,哈希映射中的键值对中的键为元素、值为该元素出现次数。 知识点&#xff1…

腾讯云2023年双11优惠活动和9999元代金券领取规则

2023腾讯云双11优惠价格表终于来了,轻量2核2G3M云服务器88元一年、轻量2核4G5M服务器166.6元一年、3年轻量2核2G4M带宽优惠价366.6元、3年轻量2核4G5M配置566.6元,CVM云服务器2核2G配置SA2实例172.3元一年、标准型S5服务器2核2G配置280.8元一年&#xff…

单目3D目标检测 方法综述——直接回归方法、基于深度信息方法、基于点云信息方法

本文综合整理单目3D目标检测的方法模型,包括:基于几何约束的直接回归方法,基于深度信息的方法,基于点云信息的方法。万字长文,慢慢阅读~ 直接回归方法 涉及到模型包括:MonoCon、MonoDLE、MonoFlex、CUPNet…

大模型在百度智能问答、搜索中的应用

本文主要介绍了智能问答技术在百度搜索中的应用。包括机器问答的发展历程、生成式问答、百度搜索智能问答应用。欢迎大家加入百度搜索团队,共同探索智能问答技术的发展方向,文末有简历投递方式。 01 什么是机器问答 机器问答,就是让计算机…

如何利用数字化转型升级,重塑企业核心竞争力?

工程机械行业是一个周期性明显的行业,企业经营受到宏观经济与国家基础设施建设的影响较大,例如企业经济上行时,加大投资扩大生产规模,以满足市场需求的增长,当经济下行时,企业可能面临减产和裁员等问题&…

【Ansible自动化运维工具 第一部分】Ansible常用模块详解(附各模块应用实例和Ansible环境安装部署)

Ansible常用模块 一、Ansible1.1 简介1.2 工作原理1.3 Ansible的特性1.3.1 特性一:Agentless,即无Agent的存在1.3.2 特性二:幂等性 1.4 Ansible的基本组件 二、Ansible环境安装部署2.1 安装ansible2.2 查看基本信息2.3 配置远程主机清单 三、…

Java练习题-输出二维数组对角线元素和

✅作者简介:CSDN内容合伙人、阿里云专家博主、51CTO专家博主、新星计划第三季python赛道Top1🏆 📃个人主页:hacker707的csdn博客 🔥系列专栏:Java练习题 💬个人格言:不断的翻越一座又…

DAY34 860.柠檬水找零 + 406.根据身高重建队列

860.柠檬水找零 题目要求:在柠檬水摊上,每一杯柠檬水的售价为 5 美元。 顾客排队购买你的产品,(按账单 bills 支付的顺序)一次购买一杯。 每位顾客只买一杯柠檬水,然后向你付 5 美元、10 美元或 20 美元…

优化预算管理流程:Web端实现预算编制的利器

本文由葡萄城技术团队原创并首发。转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具、解决方案和服务,赋能开发者。 前言:什么是预算和预算编制 预算 预算是企业在预测、决策的基础上,以数量和金额的形式…

(免费领源码)php#Thinkphp#MYSQL校园二手交易app 99211-计算机毕业设计项目选题推荐

目 录 摘要 Abstract 1 绪论 1.1 研究背景 1.2国内外研究现状 1.3论文结构与章节安排 2 校园二手物品交易app系统分析 2.1 可行性分析 2.2 系统流程分析 2.2.1 数据流程 3.3.2 业务流程 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析…

用傲梅分区软件分割分区重启系统蓝屏BAD_SYSTEM_CONFIG_INFO,八个解决参考方案

环境: Win11 专业版 HP 笔记本 傲梅分区软件 闪迪16G U盘 Win10 官方镜像文件 Win11PE 系统安装U盘 USB固态硬盘盒 问题描述: 起因 开始使用windows自动磁盘管理工具压缩不了磁盘,提示无法将卷压缩到超出任何不可移动的文件所在点,关闭系统保护还原,删除系统创建…

J41H-40C法兰截止阀型号解析

J41H-40C型号字母含义解析 J41H-40C型号是德特森阀门常用的截止阀型号字母代表的意思是: J——代表阀门类型《截止阀》 4——代表连接方式《法兰》 1——代表结构形式《直流》 H——代表阀座堆焊《不锈钢》 -《分隔键》 40——代表公称压力《4.0MPA》 C——代表阀体材料…

win操作系统下安装Deepspeed

1. 准备cunda环境 下载cuda、cudnn, CUDA Toolkit Archive | NVIDIA Developerhttps://developer.nvidia.com/cuda-toolkit-archive Log in | NVIDIA Developerhttps://developer.nvidia.com/rdp/cudnn-download 选择版本可以参考里的Index of /anaconda/cloud/p…

通过阿里云创建accessKeyId和accessKeySecret

我们想实现服务端向个人发送短信验证码 需要通过accessKeyId和accessKeySecret 这里可以白嫖阿里云的 这里 我们先访问阿里云官网 阿里云地址 进入后搜索并进入短信服务 如果没登录 就 登录一下先 然后在搜索框搜索短信服务 点击进入 因为我也是第一次操作 我们一起点免费开…

禁止U盘拷贝-保护数据安全

禁止U盘拷贝-保护数据安全 下载使用安企神U盘管理系统 U盘是一种可移动存储设备,可以方便地实现数据的传输和共享。然而,U盘的通用性和便携性也给它带来了安全隐患。如果U盘落入恶意用户手中,可能会导致敏感数据的泄露,给企业和…

在线数据库设计工具Itbuilder

有人说——不具备全局观的设计者想着如果把一片树叶设计的非常美,而忘了树叶需要树枝,树枝是构成大树的一部分,所以说去培育一片森林,才叫做设计。而数据库设计亦是如此,如果纠结于某个列某个表的设计,而忘…

01-windows栈溢出

简单栈溢出 测试环境: win xp sp3 cn 辅助环境:mac,安装了pwntoosl、msf 使用0day安全中的随书文件:0day\02栈溢出原理与实践\2_4_overflow_code_exec\Debug\stack_overflow_exec.exe git clone https://github.com/jas502n/0da…