【设计模式-1】UML和设计原则

说明:设计模式(Design Pattern)对于软件开发,简单来说,就是软件开发的套路,固定模板。在学习设计模式之前,需要首先学习UML(Unified Modeling Language,统一建模语言)和面向对象设计的七大原则。

UML

设计模式在代码开发之前,先需要使用设计图的方式表现出来,UML是生产设计图的语言,也就是模型语言。UML的结构有视图(View)、图(Diagram)、模型元素(Model Element)、通用机制(General Mechanism),这里不做过深的介绍,只介绍类图和类之间的关系。

类图(Class Diagram)

结构

一般来说,一个类由三个部分组成:类名、属性和方法。如下这个类图:
在这里插入图片描述

  • 类名:Teacher;
  • 属性:name(String)、age(int,默认值是0)、gender(String);
  • 方法:setAge(int age)、getName();

其中,属性和方法前面的符号,分别表示可见性:公有(public)、私有(private)、受保护(protected)分别用“+”、“-”、“#”来表示,在Java中,还有一种包内可见性(package),使用“*”来表示。

对于属性,在属性类型后面加 “= xx”,表示此属性设置默认值为“xx”,表示方式如下:

可见性 名称:类型 [= 默认值]

对于方法,表示方式如下:

可见性 名称([参数列表])[:返回类型]

以上方括号中的部分,表示可有可无;


如果该类是一个接口,类图表示如下:

在这里插入图片描述

其他内容不变,类名上面加两个尖括号包裹的interface,也有在类名左上角使用一个类似圆形取卡针标志来表示的,如下:

在这里插入图片描述


类之间关系

类与类之间的关系,分为以下几种:

关联关系

关联,指类与类之间存在联系,可分为双向关联、单向关联、自关联、多重性关联、聚合和组合,如下:


在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


如下,表示一个按钮(Button)只能属于一个表单(Form),是一对一的关系,而一个表单,可以有0个或多个按钮;
在这里插入图片描述

常见的多重性关系有:1…1(一对一)、0…* (0对多) 、1…*(1对多)、0…1(0对一)、m…n(多对多);


例如,汽车(Car)与发动机(Engine),发动机是汽车的组成部分,可以独立于汽车存在,并不依赖于汽车;
在这里插入图片描述


例如,头(Head)与眼睛(Eye),眼睛是头的组成部分,但是眼睛不能单独出来,头没有了眼睛也就没有了;
在这里插入图片描述

依赖关系;

例如,驾驶员(Driver)与车(Car)的关系,驾驶员依赖于车,没有车就无法使用驾驶方法;
在这里插入图片描述

泛化关系;

泛化关系,就是继承关系,如下,学生类(Student)和教师类(Teacher)继承于人类(Person);
在这里插入图片描述

实现关系;

实现关系,顾名思义,如下,船(Ship)与车(Car)实现于交通工具接口(Vehicle),并实现其移动方法(move);

在这里插入图片描述

其他

实际上,UML的图远不止这些,还有用例图(Use Case Diagram)、对象图(Object Diagram)、包图(Package Diagram)等加上类图共13个,但是类图用的最多。

在我手边的《设计模式》(第二版,清华大学出版社,第5页,刘伟主编)中,作者引用了Martin Fowler著作中的一段话,“If someone were to come up to you in a dark alley and say, ‘Psst, wanna see a UML diagram?’ that diagram would probably be a class diagram.The majority of UML diagrams Isee are class diagrams.”(“如果有人在黑暗的小巷中向你走来并对你说:'嘿,想不想看一张UML图?'那么这张图很有可能就是一张类图,我所见过的大部分的UML图都是类图”)


面向对象设计原则

面相对象设计原则,指在程序设计时遵循的规范,有以下七个:

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

单一职责原则定义:一个对象应该只包含单一的职责,并且该职责被完整地封装在一个类中。

简单来说,就是一个类,应该只干一件事。例如在没有三层架构前,一个请求,从接收、分析、处理,都在一个类里完成,无论这个请求的哪一个环节发生了改变都需要修改代码,可维护性差,代码耦合高。


开闭原则(Open-Closed Principle,OCP)

开闭原则定义:一个软件实体应当对扩展开发,对修改关闭。

就是说,后面如果需要新增需求,可以在不修改源码的基础上,对系统扩展。举个例子,一个管理系统的界面,有许多的记录,许多筛选项,最开始记录的字段少,筛选项也少,后面字段越来越多,需要新增一些筛选条件。

不符合开闭原则的做法:对DTO对象不断地新增筛选项,name(姓名)、age(年龄)、createTime(创建时间)……;

符合开闭原则的做法:在DTO对象里定义一个Map对象conditions,用于接收前端传递的条件参数。这样无论后面新增多少个筛选条件,实体类都不需要修改,只需要在Mapper.xml里面新增筛选字段(conditions.字段名)的查询条件即可;

private Map<String, String> conditions;

里氏代换原则(Liskov Substitution Principle,LSP)

里氏代换原则可以通俗表述为:在软件中如果能够使用基类(父类)对象(的地方),那么一定能够使用其子类对象。


依赖倒转原则(Dependency Inversion Principle,DIP)

依赖倒转原则定义:高层模块不应该依赖底层模块,他们都应该依赖抽象。抽象不应该依赖细节,细节应该依赖于抽象。


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

接口隔离原则定义:客户端不应该依赖那些它不需要的接口。

就是说,对于接口应该尽可能细化,而不是把一堆接口放到到一个大接口里面。如接口A表示飞行,接口B表示跳跃,接口C表示鸣叫,虽然这三个特性麻雀完全符合,但是不能把这三个接口合并到一个接口D中,然后让麻雀来实现,而是应该隔离开,让具备某种能力的类实现指定的接口。


合成复用原则(Composite Reuse Principle,CRP)

合成复用原则定义:尽量使用对象组合,而不是继承来达到复用的目的。

简单来说,就是类之间的关系,少用继承,尽量使用接口实现或者依赖注入的方式。例如有一个面试题,如何解决ArrayList线程不安全的问题,我所知道的有以下5个方法:

(1)使用CopyOnWriteArrayList();

(2)使用Collections.synchronizedList();

(3)定义一个类MyArrayList,继承ArrayList,重写其方法,每个方法用synchronized修饰;

(4)定义一个类MyArrayList,类里面定义一个ArrayList,自定义List的增删改查方法,用synchronized修饰,方法里面调用ArrayList对应的方法;

(5)使用Vector;

重点是第三个、第四个方法的对比,前者是继承,需要重写父类的方法,后者是设值注入,调用ArrayList的方法,显然后者更加灵活。


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

迪米特法则,又称最少知识原则原则,简单来说,就是指一个软件实体应当尽可能少地与其他实体发生相互作用。这样,当一个模块修改时,就会尽量少地影响其他的模块。

总结

本篇文章内容基本来自《设计模式》(第2版,清华大学出版社,刘伟主编)第1、2章

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

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

相关文章

vueday02——使用NTableData

1.下载naivueui 2.按需导入&#xff0c;不要全局导入 注意不要导入错误组件或者写错组件名称 import { NDataTable } from naive-ui 3.定义表头和数据&#xff01;&#xff01;&#xff01; n-data-table标签必须要使用数据和数据 少一个都不能正确渲染&#xff01;&#xf…

CSS阶详细解析一

CSS进阶 目标&#xff1a;掌握复合选择器作用和写法&#xff1b;使用background属性添加背景效果 01-复合选择器 定义&#xff1a;由两个或多个基础选择器&#xff0c;通过不同的方式组合而成。 作用&#xff1a;更准确、更高效的选择目标元素&#xff08;标签&#xff09;。…

Web3 整理React项目 导入Web3 并获取区块链信息

上文 WEB3 创建React前端Dapp环境并整合solidity项目&#xff0c;融合项目结构便捷前端拿取合约 Abi 我们用react 创建了一个 dapp 项目 并将前后端代码做了个整合 那么 我们就来好好整理一下 我们的前端react的项目结构 我们在 src 目录下创建一个 components 用来存放我们的…

哨兵1号后向散射系数土壤水分反演

哨兵1号后向散射系数土壤水分反演 数据导入 打开之前预处理之后的VH和VV极化的后向散射系数转存的tiff文件 导入实测点 选择KML转图层 kml文件是由奥维地图导出的.ovkml格式改后缀名得到的 提取采样点的后向散射系数 选择多值提取至点 右键打开点图层的属性表,发现…

k8s-18 认证授权

Authentication (认证) 认证方式现共有8种&#xff0c;可以启用一种或多种认证方式&#xff0c;只要有一种认证方式通过&#xff0c;就不再进行其它方式的认证。通常启用X509 Client Certs和Service Accout Tokens两种认证方式 Kubernetes集群有两类用户:由Kubernetes管理的Ser…

Oracle database 开启归档日志 archivelog

Oracle database 开启归档日志 archivelog 归档日志模式 (Archivelog Mode)。归档日志模式是一种数据库运行模式&#xff0c;它允许数据库将日志文件保存到归档日志目录中&#xff0c;以便在需要时进行恢复和还原操作。通过开启归档日志模式&#xff0c;可以提高数据库的可靠性…

服务器中了locked勒索病毒怎么办,勒索病毒解密,数据恢复

最近一段时间内&#xff0c;相信很多使用金蝶或用友的办公软件的企业&#xff0c;有很多都经历了locked勒索病毒的攻击&#xff0c;导致企业服务器被加密无法正常使用&#xff0c;严重影响了企业的正常工作。通过云天数据恢复中心的解密恢复发现&#xff0c;在今年locked勒索病…

缓解Oracles数据库内存不足的问题

解决思路&#xff0c;调整内存参数&#xff1b; 安装Oracle时&#xff0c;为了均衡电脑性能和数据库性能&#xff0c;Oracle一个实例默认内存占用大小为物理内存的1/8。 如环境不需要分配那么大的内存来支撑Oracle&#xff0c;可通过修改 sga_max_size 的值来减少系统中内存占…

STROBE-MR

Welcome to the STROBE-MR website! About: STROBE-MR stands for “Strengthening the Reporting of Observational Studies in Epidemiology using Mendelian Randomization”. Inspired by the original STROBE checklist, the STROBE-MR guidelines were developed to ass…

C++_linux下_非阻塞键盘控制_程序暂停和继续

1. 功能 在程序执行过程中,点击键盘p按键(pause), 程序暂停, 点击键盘上的n按键(next),程序继续执行 2. 代码 #include <iostream> #include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <sys/ioctl.h> char get_keyboar…

通信设备为什么需要接地阻?

在现代社会&#xff0c;通信设备已经成为我们生活中不可或缺的一部分。无论是手机、电视、计算机还是互联网路由器&#xff0c;它们都在不断地演进&#xff0c;以满足我们日益增长的通信需求。 实际应用中&#xff0c;我们都会了解到通信设备在正常运行时产生的电磁波辐射和电…

改写二分搜索算法

描述 设a[0:n-1]是已排好序的数组&#xff0c;请改写二分搜索算法&#xff0c;使得当x不在数组中时&#xff0c;返回小于x的最大元素位置i和大于x的最小元素位置j。当搜索元素在数组中时&#xff0c;i和j相同&#xff0c;均为x在数组中的位置。 输入格式: 输入有两行&#x…

解决报错:RuntimeError: “LayerNormKernelImpl“ not implemented for ‘Half‘

文章目录 1. 为什么报错&#xff1f;2. 解决办法2.1 方法12.2 方法22.3 其他 1. 为什么报错&#xff1f; 一般发生在模型推理过程中&#xff0c;由于精度导致的报错&#xff0c;一些硬件和框架对于半精度操作的支持可能有限&#xff0c;导致无法执行特定的操作。 2. 解决办法…

Flink之转换算子Transformation

转换算子Transformation 概述基本转换算子映射Map扁平映射flatMap过滤Filter 聚合算子按键分区keyBy归约聚合reduce简单聚合sum、min、max、minBy、maxBy 物理分区算子随机分配轮询分配重缩放广播全局分区自定义分区 分流操作Filter分流SideOutPut分流Split分流 合流操作联合Un…

4K壁纸小程序源码 全内容自动采集

全内容自动采集 4K壁纸小程序源码&#xff0c;带流量主。用的都是一个接口&#xff0c;不过这个不知是谁改的&#xff0c;成了LSP版&#xff0c;是真色啊&#xff0c;专搜小姐姐。 4K壁纸&#xff0c;静态壁纸&#xff0c;头像等都有保留&#xff0c;界面广告位很多&#xff0c…

线性回归原理

1、 线性回归的原理 1.1 线性回归应用场景 房价预测 销售额度预测 金融&#xff1a;贷款额度预测、利用线性回归以及系数分析因子1.2 什么是线性回归 1.2.1定义与公式 线性回归(Linear regression)是利用回归方程(函数)对一个或多个自变量(特征值)和因变量(目标值)之间关系…

导航守卫的使用记录和beforeEach( )死循环的问题

前置导航守卫beforeEach的使用 import Vue from vue import VueRouter from vue-router // 进度条 import NProgress from nprogress import nprogress/nprogress.cssVue.use(VueRouter)// 路由表 const routes [{path: "/",redirect: "/home",},{path: …

Unity3D Shader新手入门教程:3D溶解与腐蚀特效详解

引言 在游戏开发中&#xff0c;特效是非常重要的一部分&#xff0c;它能够增加游戏的趣味性和可玩性。其中&#xff0c;Shader特效是一种非常常见和常用的特效&#xff0c;它能够通过改变物体表面的渲染方式来实现各种各样的特效效果。本文将详细介绍Unity3D中的Shader 3D溶解与…

04 MIT线性代数-矩阵的LU分解 Factorization into A=LU

目的: 从矩阵的角度理解高斯消元法, 完成LU分解得到ALU 1.矩阵乘积的逆矩阵 Inverse of a product 2.矩阵乘积的转置 Transpose of a product 3.转置矩阵的逆矩阵 Inverse of a transpose 4.矩阵的LU分解 U为上三角阵(Upper triangular matrix), L为下三角阵(Lower triangular…

Qt系列-常用控件使用整理

1、QMainWindow介绍 菜单栏最多只有一个 //菜单栏创建 菜单栏最多只能有一个QMenuBar*bar menuBar();//将菜单栏放入到窗口中setMenuBar(bar);//创键菜单QMenu*fileMenubar->addMenu("文件");QMenu*editMenubar->addMenu("编辑");//创建菜单项QActi…