换个思路快速上手UML和plantUML——时序图

2024-01-30 01:36

本文主要是介绍换个思路快速上手UML和plantUML——时序图,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一章我们介绍了类图,我们很清楚,类图是从更加宏观的角度去梳理系统结构的,从类图中我们可以获取到类与类之间:继承,实现等关系信息,是宏观逻辑。下面我们继续换一个思路:作为一名软件工程结构化图的设计者去设计另一种图,要求:

1)这种图要微观的体现调用关系

2)要线性的体现调用的时间关系

3)要能体现不同逻辑的不同结果

而这种图主要用于阅读源码,或者向别人介绍代码思路的等等相关的场景。

在这里插入图片描述

1.入手代码

对比设计类图,这个微观一点图显然要更加复杂,为了能成功把这种图设计出来,所以我们从实际的一段代码出发

package se;
/*** @author: Jeffrey* @date: 2024/01/29/11:06* @description:*/
public class TimeSequenceUML {public static void main(String[] args) {Client client = new Client();client.work();}
}
class Device{public void write(String hello) {}
}
class Server{private Device device;public void close() {}public void open() {}public void print(String hello) {device.write(hello);}
}class Client{private Server server;public void work(){server.open();server.print("hello");server.close();}
}

2.代码分析

2.1 图的元素

这段代码并不难理解,调用关系也并不复杂,如果直接去像我们平时阅读源码的思路一样去走方法,可以得到一个线性的调用思路,但是我们经常会遇到这么几个问题:

  1. 代码的核心对象是谁?
  2. 代码的调用者是谁?
  3. 都是对象,这些对象的之间各自承担的逻辑任务其实很容易搞混

说白了:为了设计好这个图,我们要完成的第一件事就是把这些代码的元素抽象出来,用一个图形表示就好了。于是优秀的我们思考了很多相关的对象完成以下的元素设计:

元素:含义:图像:plantUml:
角色一般是逻辑的开始交互者,可以是:人,机器,系统image-20240129115711016actor name
对象一般是逻辑的中间参与者,一般代表对象,又可以改变形状为以下具体的类型image-20240129115932546participant client
实体image-20240129120139844entity name
控制image-20240129120208121control name
数据库image-20240129120216366database name
集合image-20240129120222746collections name
队列image-20240129120228971queue name
边界image-20240129120235244boundary name

2.2 图的结构

将对象抽象出来后,下一步我们必须要考虑的就是如何将这些对象组织起来:

学习过线程进程,操作系统,计算机体系结构中任何一门课程的朋友们都知道:我们感觉计算机的某个操作是一瞬间的事情,其实它经历了很多个线性阶段,即使是多核多处理器的计算机,也是由一个个线性的阶段和并行的阶段链接而成,如何让我们简单明了的直接看图就能看懂程序?那么就要在图中体现程序的线性和并行关系

可是连接起来之后,不得不考虑的一个问题就是元素和元素要怎么摆放?如何才能同时体现一段时间内在调用时间上的线性特征一个瞬间时方法调用的顺序线性特征

聪明的我们明白,是时候定下一些规则来了!

  1. 关系越紧密,交互越密集的元素要尽可能的放到越近的地方
  2. 我们定义一个坐标轴,纵向代表时间顺序的线性关系
  3. 横向代表一瞬间方法调用的线性关系,或者说方法调用的层次关系

image-20240129160639721

2.3 消息和生命周期

看看我们现在有什么?有元素,有规则,下一步就是把调用顺序放进去了!记得黑盒模型吗?我们知道封装的思想有一部分就是为了隐藏方法的具体实现细节,我们现在主要关注的是方法的顺序,我们可以把每个调用过程当作是一个黑盒,只关注黑盒的输入输出信息。

我们用:

image-20240129161048506

实线实箭头来表示调用,用虚线实箭头来表示返回信息

在PlantUML中:
-> 实线
--> 虚线

image-20240129161720713

消息已经做好了,回到我们的需求,我们需要反应每个对象在什么阶段是激活的,在什么时候就跟这个对象没有什么关系了。

但是这里我们要注意一个区别,对于Java 来说,大家都知道基于JVM,对象的回收一般是不由我们来负责的【我们负责的主要是一些系统的接口或者说一些通道/流的开关】,而在UML中也类似,对象的激活状态代表的不是如IOC中对象一直的在内存中/在容器中,而是代表:在一个对象工作的同时,另一个对象是否在工作?如果在工作,表示为激活,反之为不活跃。

也就是说:我们要反映同一时刻对象的活跃状态

考虑到生命周期和时间是线性相关的,我们用在对象上的长矩形来体现生命周期的长短

image-20240129162448097

用横向上的矩形在同一水平来表示同时刻不同对象的活跃状态

image-20240129162726726

进一步的完善我们的图:

image-20240129162851557

2.4补充语法

至此我们已经能基本完成一个简单代码的线性过程,为了方便我们的绘图和绘制更加直观的图,我们补充一部分PlantUML的语法:

2.4.1 组合片段
ALT

抉择组合片段 ALT:说白了就是要在图中体现IF-ELSE 逻辑
比如这串代码:

package se;/*** @author: Jeffrey* @date: 2024/01/23/21:55* @description:*/
public class Test {public static void main(String[] args) {Student student = new Student();if (student.isGender()){Mapper mysql = new MySqlMapper();mysql.insert(student);}else {Mapper sqlServer =new SqlserverMapper();sqlServer.insert(student);}}
}
class Student{private boolean gender = true;public boolean isGender() {return gender;}
}
interface Mapper{public void insert(Student student);
}
class MySqlMapper implements Mapper{@Overridepublic void insert(Student student) {//持久化到Mysql}
}
class SqlserverMapper implements Mapper{@Overridepublic void insert(Student student) {//持久化到SqlServer}
}

可以绘制为:

image-20240129165049761

如果用plant UML 来表示:

@startuml
actor wo
Participant Test
Participant Student
Participant MySqlMapper
Participant SqlserverMapperautonumber
wo -> Test : main()
Test -> Student: isGender()
activate Student
alt Gender == trueStudent -> MySqlMapper:insert()activate MySqlMapperMySqlMapper --> Studentdeactivate MySqlMapper
elseStudent -> SqlserverMapper:insert()activate SqlserverMapperSqlserverMapper --> Studentdeactivate SqlserverMapper
end
Student --> Test
deactivate Student
@enduml

补充:

  1. 在plantUML 中if-else 逻辑通过 alt /else 来表示,如果需要else if,只需要再添加一个else,同时这个组合片段必须用end 来结尾
  2. 一般当向一个对象发出消息时,这个对象被激活,进入生命周期,返回消息时被置为不活跃状态,所以我在发完消息后用activate name 来激活对象,在回复完消息后用 deactivate name
  3. autonumber 可以用于自动对消息进行编号
  4. 还可以在第一条消息发送前用autoactivate on 来自动填充activate,然后每次需要返回消息时用return 代替:

综上可优化为:

autonumber
autoactivate on
wo -> Test : main()
Test -> Student: isGender()
activate Student
alt Gender == trueStudent -> MySqlMapper:insert()return
elseStudent -> SqlserverMapper:insert()return
end
return
Loop

显然这可用于代表循环

loop 1000 timesStudent -> SqlserverMapper:insert()returnend

image-20240129170046241

Group

可直接简单用于将一部分逻辑框起来,再添加对应的注释

    group SqlServer逻辑单元 [数据持久化]Student -> SqlserverMapper:insert()returnend

image-20240129170412126

2.4.2 逻辑分割/分隔符

可以将一块代码分割成若干逻辑模块/或者代码分层:

wo -> Test : main()
Test -> Student: isGender()
==初始化==
alt Gender == trueStudent -> MySqlMapper:insert()return
elsegroup SqlServer逻辑单元 [数据持久化]Student -> SqlserverMapper:insert()returnend
end
==持久层==

image-20240129170653321

2.4.3 让响应信息显示在箭头下面

可以使用skinparam responseMessageBelowArrow true命令,让响应信息显示在箭头下面。

2.4.4 增加彼此空间

可以使用|||来增加空间。

@startuml
Alice -> Bob: message 1
Bob --> Alice: ok
|||
Alice -> Bob: message 2
Bob --> Alice: ok
||45||
Alice -> Bob: message 3
Bob --> Alice: ok
@enduml

image-20240129170856693

2.4.5 包裹参与者

可以使用boxend box画一个盒子将参与者包裹起来。

2.4.5 并行

可以使用parend par 来将同一时间段并行逻辑括起来

3.代码实战

我们用之前我写的一个简单Netty 实现的servlet 容器的代码来进行一次简单的项目实战,源码地址是:

Netty实现简易tomcat容器

大概整体项目并不规范的时序思路是这样的:

img

这个项目实现的基本功能就是根据自定义协议传入网页请求信息,实现对信息的解码,并包装成实现了HttpServlet 接口的容器。

而我们的目标就是用时序图来完成从HelloServlet收到一个登录请求的调用过程的绘图:

业务层

@startuml
autonumber
autoactivate on
Actor browser as br
Participant HelloServlet as hs
Participant HttpServletRequest as req
Participant UserService as us
br -> hs:dePost(req,rep)
hs->req:getParameter(username)
return String:username
hs->req:getParameter(password)
return String:password
hs->us:login(username,password)
return false
return response
@enduml

image-20240129222907980

本篇关键词:时间顺序,方法调用顺序,生命周期=激活状态

这篇关于换个思路快速上手UML和plantUML——时序图的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/658730

相关文章

电脑桌面文件删除了怎么找回来?别急,快速恢复攻略在此

在日常使用电脑的过程中,我们经常会遇到这样的情况:一不小心,桌面上的某个重要文件被删除了。这时,大多数人可能会感到惊慌失措,不知所措。 其实,不必过于担心,因为有很多方法可以帮助我们找回被删除的桌面文件。下面,就让我们一起来了解一下这些恢复桌面文件的方法吧。 一、使用撤销操作 如果我们刚刚删除了桌面上的文件,并且还没有进行其他操作,那么可以尝试使用撤销操作来恢复文件。在键盘上同时按下“C

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

hdu 4565 推倒公式+矩阵快速幂

题意 求下式的值: Sn=⌈ (a+b√)n⌉%m S_n = \lceil\ (a + \sqrt{b}) ^ n \rceil\% m 其中: 0<a,m<215 0< a, m < 2^{15} 0<b,n<231 0 < b, n < 2^{31} (a−1)2<b<a2 (a-1)^2< b < a^2 解析 令: An=(a+b√)n A_n = (a +

v0.dev快速开发

探索v0.dev:次世代开发者之利器 今之技艺日新月异,开发者之工具亦随之进步不辍。v0.dev者,新兴之开发者利器也,迅速引起众多开发者之瞩目。本文将引汝探究v0.dev之基本功能与优势,助汝速速上手,提升开发之效率。 何谓v0.dev? v0.dev者,现代化之开发者工具也,旨在简化并加速软件开发之过程。其集多种功能于一体,助开发者高效编写、测试及部署代码。无论汝为前端开发者、后端开发者

三相直流无刷电机(BLDC)控制算法实现:BLDC有感启动算法思路分析

一枚从事路径规划算法、运动控制算法、BLDC/FOC电机控制算法、工控、物联网工程师,爱吃土豆。如有需要技术交流或者需要方案帮助、需求:以下为联系方式—V 方案1:通过霍尔传感器IO中断触发换相 1.1 整体执行思路 霍尔传感器U、V、W三相通过IO+EXIT中断的方式进行霍尔传感器数据的读取。将IO口配置为上升沿+下降沿中断触发的方式。当霍尔传感器信号发生发生信号的变化就会触发中断在中断

Jenkins 插件 地址证书报错问题解决思路

问题提示摘要: SunCertPathBuilderException: unable to find valid certification path to requested target...... 网上很多的解决方式是更新站点的地址,我这里修改了一个日本的地址(清华镜像也好),其实发现是解决不了上述的报错问题的,其实,最终拉去插件的时候,会提示证书的问题,几经周折找到了其中一遍博文

利用Django框架快速构建Web应用:从零到上线

随着互联网的发展,Web应用的需求日益增长,而Django作为一个高级的Python Web框架,以其强大的功能和灵活的架构,成为了众多开发者的选择。本文将指导你如何从零开始使用Django框架构建一个简单的Web应用,并将其部署到线上,让世界看到你的作品。 Django简介 Django是由Adrian Holovaty和Simon Willison于2005年开发的一个开源框架,旨在简

CentOs7上Mysql快速迁移脚本

因公司业务需要,对原来在/usr/local/mysql/data目录下的数据迁移到/data/local/mysql/mysqlData。 原因是系统盘太小,只有20G,几下就快满了。 参考过几篇文章,基于大神们的思路,我封装成了.sh脚本。 步骤如下: 1) 先修改好/etc/my.cnf,        ##[mysqld]       ##datadir=/data/loc

SAM2POINT:以zero-shot且快速的方式将任何 3D 视频分割为视频

摘要 我们介绍 SAM2POINT,这是一种采用 Segment Anything Model 2 (SAM 2) 进行零样本和快速 3D 分割的初步探索。 SAM2POINT 将任何 3D 数据解释为一系列多向视频,并利用 SAM 2 进行 3D 空间分割,无需进一步训练或 2D-3D 投影。 我们的框架支持各种提示类型,包括 3D 点、框和掩模,并且可以泛化到不同的场景,例如 3D 对象、室

UE5 半透明阴影 快速解决方案

Step 1: 打开该选项 Step 2: 将半透明材质给到模型后,设置光照的Shadow Resolution Scale,越大,阴影的效果越好