【设计模式-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章
相关文章:

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

【数据结构】——二叉树的基础知识
数概念及结构 数的分类 二叉树、多叉树 数的概念 树是一种非线性的数据结构,它是由n(n>0)个有限节点组成一个具有层次关系的集合。把它叫做树的原因是它看起来像一颗倒挂的树,也就是说它是跟朝上,而叶朝下的。 有一个特殊的节点&…...
日常bug汇总
1.constraintlayout NestedScrollView 可能会导致NestedScrollView 不滑动 2.截屏 open class SecureFragment : LogLifecycleFragment() {override fun onResume() {super.onResume()if (!BuildConfig.DEV) {requireActivity().window.setFlags(WindowManager.LayoutParam…...

C#使用PPT组件的CreateVideo方法生成视频
目录 需求 实现 CreateVideo方法 关键代码 CreateVideoStatus 其它 需求 我们在使用PowerPoint文档时,经常会使用其导出功能以创建视频,如下图: 手工操作下,在制作好PPT文件后,点击文件 -> 导出 -> 创建视…...

数字化体系如何帮助企业拓展裂变增长渠道?数字化营销体系构建?
在当前的商业环境中,数字不仅仅是数据,还代表着技术和资产。企业数字化正是将数据转化为资产的过程。从信息化时代到数字化时代,企业逐渐将业务和组织、管理和创收都朝着在线化和数据化的方向发展,特别是企业的业务板块。数字化营…...

关于vant 的tabbar功能
1、想要实现tabbar页面A,其他的页面B(非tabbar页面)。 从A页面进入B页面,底部的active选中效果应该被取消掉,但是还是选中A。 按照官网的说法有两个方法 一、根据path路径 二、自定义的model 但是!但是…...
:style动态绑定,但只要页面发生变化就会执行一次方法
1、问题 开发过程中有个需求是遍历列表绘制div,div的样式是后端接口传来的,一开始这种写法,:style“formatStyle(item)”,写在了模板中 这样写发现一个问题,只要页面发生重绘,比如页面输入框输入数字&…...

文件的逻辑结构(顺序文件,索引文件)
所谓的“逻辑结构”,就是指在用户看来,文件内部的数据应该是如何组织起来的。 而“物理结构”指的是在操作系统看来,文件的数据是如何存放在外存中的。 1.无结构文件 无结构文件:文件内部的数据就是一系列二进制流或字符流组成。无明显的逻…...

suricata匹配从入门到精通(五)----二次开发保护规则库
0x00 背景 开源的suricata资源包是没有做加密处理,如果想要保护资源包,需要二次开发修改suricata源码。 本文基于suricata6.0.1 版本https://github.com/OISF/suricata/archive/refs/tags/suricata-6.0.1.zip二开。 0x01 实践 通过debug,跟规则处理相关需要修改2个地方。…...

软件测试肖sir__python之ui自动化定位方法(2)
Selenium中元素定位方法 一、定位方法 要实现UI自动化,就必须学会定位web页面元素,Selenium核心 webdriver模块提供了9种定位元素方法: 定位方式 提供方法 id定位 find_element_by_id() name定位 find_element_by_name() class定位 find_elem…...

【JVM面试题】JVM分代年龄为何是15次?能设置为16吗?
系列文章目录 【JVM系列】第一章 运行时数据区 【JVM面试题】第二章 从JDK7 到 JDK8, JVM为啥用元空间替换永久代? 【JVM面试题】第三章 JVM分代年龄为何是15次?能设置为16吗? 大家好,我是青花。拥有多项发明专利(都是…...

java三层架构/表现层-业务层-持久层
三层架构 什么是 Java 三层架构 三层架构是指:视图层view(表现层),服务层service(业务逻辑层),持久层Dao(数据访问层), Java的三层架构是指将Java程序分为三…...

视频监控这样做,简单又高效!
随着技术的不断进步,视频监控系统已经变得更加高效和智能,可以提供更全面的监控和分析功能,有助于提高安全性、管理效率和决策支持。 客户案例 超市连锁店 福建某全国性超市连锁店面临高额商品损失、偷窃问题,以及对客户安全和员工…...
掌握 Go 的计时器
简介 定时器是任何编程语言的重要工具,它允许开发人员在特定时间间隔安排任务或执行代码。在 Go 中,定时器是通过 time 包实现的,该包提供了一系列功能来创建、启动、停止和有效处理定时器。我们将探索 Go 中定时器的强大功能,并…...

嵌入式软件开发笔试面试
C语言部分: 1.gcc的四步编译过程 1.预处理 展开头文件,删除注释、空行等无用内容,替换宏定义。 gcc -E hello.c -o hello.i 2.编译 检查语法错误,如果有错则报错,没有错误则生成汇编文件。 gcc -S hello.i -o h…...

【Qt高阶】Linux安装了多个版本的Qt 部署Qt程序,出包【2023.10.17】
简介 linux系统下可执行程序运行时会加载一些动态库so,有一些是Qt的库,Qt的库会加载其他更基础的库。最后出包的时候需要把依赖的包整理到一个文件夹,来制作安装包。近期遇到已经将依赖的so文件拷贝至程序目录下,但还是调系统路径…...
OpenGL简介
OpenGL 本身并不是一个 API,它仅仅是一个由 Khronos组织 制定并维护的规范(Specification)。规范严格规定了每个函数该如何执行,以及它们的输出值。至于内部具体每个函数是如何实现的,将由 OpenGL 库的开发者自行决定。…...

持续集成工具jenkins操作
安装Jenkins 下载jenkins安装包 linux上下载jenkins失败 开始在windows上安装jenkins 1、先安装JDK https://jingyan.baidu.com/article/fdbd4277dd90f0b89e3f489f.html 免安装版本JDK只需要解压配置环境变量即可 2、安装Jenkins 参考文档: https://www.cnb…...

使用BurpSuite抓取HTTPS接口
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 原因设置方式 原因 BurpSuite之所以不能抓取https数据包,是因为BurpSuite作为中间人代理,我们和https网站之间的数据通信都是由BurpSuite来…...

移动硬盘被格式化了如何恢复数据?四步教你如何恢复
在日常生活中,我们常常会使用各种存储设备来保存和备份我们的重要数据。移动硬盘作为一种便携式的存储设备,被广泛应用于数据的存储和传输。然而,有时候我们会不小心将移动硬盘格式化,从而丢失了里面的数据。本文将介绍移动硬盘格…...

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?
编辑:陈萍萍的公主一点人工一点智能 未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战,在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...
[2025CVPR]DeepVideo-R1:基于难度感知回归GRPO的视频强化微调框架详解
突破视频大语言模型推理瓶颈,在多个视频基准上实现SOTA性能 一、核心问题与创新亮点 1.1 GRPO在视频任务中的两大挑战 安全措施依赖问题 GRPO使用min和clip函数限制策略更新幅度,导致: 梯度抑制:当新旧策略差异过大时梯度消失收敛困难:策略无法充分优化# 传统GRPO的梯…...
CVPR 2025 MIMO: 支持视觉指代和像素grounding 的医学视觉语言模型
CVPR 2025 | MIMO:支持视觉指代和像素对齐的医学视觉语言模型 论文信息 标题:MIMO: A medical vision language model with visual referring multimodal input and pixel grounding multimodal output作者:Yanyuan Chen, Dexuan Xu, Yu Hu…...
rknn优化教程(二)
文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK,开始写第二篇的内容了。这篇博客主要能写一下: 如何给一些三方库按照xmake方式进行封装,供调用如何按…...
基于Uniapp开发HarmonyOS 5.0旅游应用技术实践
一、技术选型背景 1.跨平台优势 Uniapp采用Vue.js框架,支持"一次开发,多端部署",可同步生成HarmonyOS、iOS、Android等多平台应用。 2.鸿蒙特性融合 HarmonyOS 5.0的分布式能力与原子化服务,为旅游应用带来…...
Golang dig框架与GraphQL的完美结合
将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用,可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器,能够帮助开发者更好地管理复杂的依赖关系,而 GraphQL 则是一种用于 API 的查询语言,能够提…...

转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...
【算法训练营Day07】字符串part1
文章目录 反转字符串反转字符串II替换数字 反转字符串 题目链接:344. 反转字符串 双指针法,两个指针的元素直接调转即可 class Solution {public void reverseString(char[] s) {int head 0;int end s.length - 1;while(head < end) {char temp …...
如何为服务器生成TLS证书
TLS(Transport Layer Security)证书是确保网络通信安全的重要手段,它通过加密技术保护传输的数据不被窃听和篡改。在服务器上配置TLS证书,可以使用户通过HTTPS协议安全地访问您的网站。本文将详细介绍如何在服务器上生成一个TLS证…...