换个思路快速上手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

相关文章

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

C++快速排序超详细讲解

《C++快速排序超详细讲解》快速排序是一种高效的排序算法,通过分治法将数组划分为两部分,递归排序,直到整个数组有序,通过代码解析和示例,详细解释了快速排序的工作原理和实现过程,需要的朋友可以参考下... 目录一、快速排序原理二、快速排序标准代码三、代码解析四、使用while循环的快速排序1.代码代码1.由快

Win32下C++实现快速获取硬盘分区信息

《Win32下C++实现快速获取硬盘分区信息》这篇文章主要为大家详细介绍了Win32下C++如何实现快速获取硬盘分区信息,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实现代码CDiskDriveUtils.h#pragma once #include <wtypesbase

Spring AI与DeepSeek实战一之快速打造智能对话应用

《SpringAI与DeepSeek实战一之快速打造智能对话应用》本文详细介绍了如何通过SpringAI框架集成DeepSeek大模型,实现普通对话和流式对话功能,步骤包括申请API-KEY、项目搭... 目录一、概述二、申请DeepSeek的API-KEY三、项目搭建3.1. 开发环境要求3.2. mav

Python如何快速下载依赖

《Python如何快速下载依赖》本文介绍了四种在Python中快速下载依赖的方法,包括使用国内镜像源、开启pip并发下载功能、使用pipreqs批量下载项目依赖以及使用conda管理依赖,通过这些方法... 目录python快速下载依赖1. 使用国内镜像源临时使用镜像源永久配置镜像源2. 使用 pip 的并

SpringBoot快速接入OpenAI大模型的方法(JDK8)

《SpringBoot快速接入OpenAI大模型的方法(JDK8)》本文介绍了如何使用AI4J快速接入OpenAI大模型,并展示了如何实现流式与非流式的输出,以及对函数调用的使用,AI4J支持JDK8... 目录使用AI4J快速接入OpenAI大模型介绍AI4J-github快速使用创建SpringBoot

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE