深入理解系统:UML类图

news2025/6/6 9:18:05

UML类图

类图(class diagram) 描述系统中的对象类型,以及存在于它们之间的各种静态关系

正向工程(forward engineering)在编写代码之前画UML图。
逆向工程(reverse engineering)从已有代码建造UML图,目的是帮助人们理解代码。

案例1:电商系统类图

在这里插入图片描述

# UML类图元素:类、属性、方法、继承、关联、聚合
from typing import List

class User:
    def __init__(self, user_id: str, name: str):
        self.user_id = user_id  # 公有属性
        self._name = name       # 保护属性
        self.__password = ""    # 私有属性
    
    def login(self, password: str) -> bool:  # 公有方法
        """验证用户登录"""
        return password == self.__password

class Customer(User):  # 继承关系 (泛化)
    def __init__(self, user_id: str, name: str):
        super().__init__(user_id, name)
        self.cart = ShoppingCart()  # 组合关系 (强拥有)
    
    def place_order(self) -> Order:
        """创建订单"""
        return Order(self, self.cart.items)

class Seller(User):  # 继承关系
    def __init__(self, user_id: str, name: str, store: Store):
        super().__init__(user_id, name)
        self.store = store  # 关联关系
        
    def add_product(self, product: Product):
        """添加商品到店铺"""
        self.store.products.append(product)

class Product:
    def __init__(self, product_id: str, name: str, price: float):
        self.product_id = product_id
        self.name = name
        self.price = price

class ShoppingCart:
    def __init__(self):
        self.items: List[Product] = []  # 聚合关系 (弱拥有)
    
    def add_item(self, product: Product):
        self.items.append(product)
    
    def calculate_total(self) -> float:
        return sum(item.price for item in self.items)

class Order:
    def __init__(self, customer: Customer, items: List[Product]):
        self.customer = customer  # 关联关系
        self.items = items
        self.status = "Pending"
    
    def process_payment(self, payment: PaymentProcessor):  # 依赖关系
        payment.process(self.calculate_total())
    
    def calculate_total(self) -> float:
        return sum(item.price for item in self.items)

class Store:
    def __init__(self, store_id: str, name: str):
        self.store_id = store_id
        self.name = name
        self.products: List[Product] = []  # 聚合关系

# 接口实现 (依赖倒置)
class PaymentProcessor(ABC):  # 抽象类/接口
    @abstractmethod
    def process(self, amount: float):
        pass

class CreditCardProcessor(PaymentProcessor):  # 实现关系
    def process(self, amount: float):
        print(f"Processing credit card payment: ${amount:.2f}")

class PayPalProcessor(PaymentProcessor):  # 实现关系
    def process(self, amount: float):
        print(f"Processing PayPal payment: ${amount:.2f}")

案例2:车辆租赁系统类图

在这里插入图片描述

# UML类图元素:抽象类、枚举、组合、聚合、依赖
from abc import ABC, abstractmethod
from enum import Enum
from datetime import date

class VehicleType(Enum):  # 枚举类
    CAR = 1
    TRUCK = 2
    SUV = 3
    MOTORCYCLE = 4

class AbstractVehicle(ABC):  # 抽象类
    def __init__(self, license_plate: str, model: str, year: int):
        self.license_plate = license_plate
        self.model = model
        self.year = year
        self.available = True
    
    @abstractmethod
    def get_rental_rate(self) -> float:
        pass

class Car(AbstractVehicle):  # 继承
    def __init__(self, license_plate: str, model: str, year: int, seats: int):
        super().__init__(license_plate, model, year)
        self.seats = seats
    
    def get_rental_rate(self) -> float:  # 实现抽象方法
        return 50.0 + (self.seats * 5)

class Truck(AbstractVehicle):  # 继承
    def __init__(self, license_plate: str, model: str, year: int, capacity: float):
        super().__init__(license_plate, model, year)
        self.capacity = capacity  # 载重能力(吨)
    
    def get_rental_rate(self) -> float:
        return 100.0 + (self.capacity * 20)

class RentalAgency:
    def __init__(self, name: str):
        self.name = name
        self.fleet: List[AbstractVehicle] = []  # 聚合
        self.rentals: List[RentalContract] = []  # 组合
    
    def add_vehicle(self, vehicle: AbstractVehicle):
        self.fleet.append(vehicle)
    
    def rent_vehicle(self, customer: Customer, vehicle: AbstractVehicle, 
                     start_date: date, end_date: date):
        if vehicle.available:
            contract = RentalContract(customer, vehicle, start_date, end_date)
            self.rentals.append(contract)
            vehicle.available = False
            return contract
        return None

class Customer:
    def __init__(self, customer_id: str, name: str):
        self.customer_id = customer_id
        self.name = name
        self.license_number = ""

class RentalContract:  # 组合类
    def __init__(self, customer: Customer, vehicle: AbstractVehicle, 
                 start_date: date, end_date: date):
        self.customer = customer
        self.vehicle = vehicle
        self.start_date = start_date
        self.end_date = end_date
        self.total_cost = self.calculate_cost()
    
    def calculate_cost(self) -> float:
        days = (self.end_date - self.start_date).days
        return days * self.vehicle.get_rental_rate()
    
    def generate_invoice(self, printer: InvoicePrinter):  # 依赖关系
        printer.print_invoice(self)

class InvoicePrinter:  # 工具类
    def print_invoice(self, contract: RentalContract):
        print(f"Invoice for {contract.customer.name}")
        print(f"Vehicle: {contract.vehicle.model}")
        print(f"Total: ${contract.total_cost:.2f}")

案例3:学校管理系统类图

在这里插入图片描述

# UML类图元素:多重继承、接口实现、依赖、关联
from abc import ABC, abstractmethod
from datetime import date

class Person:
    def __init__(self, name: str, birth_date: date):
        self.name = name
        self.birth_date = birth_date
    
    def get_age(self) -> int:
        today = date.today()
        return today.year - self.birth_date.year

class Researcher(ABC):  # 接口
    @abstractmethod
    def conduct_research(self, topic: str):
        pass

class Teacher(Person):  # 单继承
    def __init__(self, name: str, birth_date: date, department: str):
        super().__init__(name, birth_date)
        self.department = department
        self.courses: List[Course] = []  # 双向关联
    
    def assign_course(self, course: 'Course'):
        self.courses.append(course)
        course.teacher = self

class Professor(Teacher, Researcher):  # 多重继承
    def __init__(self, name: str, birth_date: date, department: str, title: str):
        Teacher.__init__(self, name, birth_date, department)
        self.title = title
    
    def conduct_research(self, topic: str):  # 实现接口
        print(f"Conducting research on {topic}")
    
    def supervise_phd(self, student: 'PhdStudent'):
        student.advisor = self

class Student(Person):
    def __init__(self, name: str, birth_date: date, student_id: str):
        super().__init__(name, birth_date)
        self.student_id = student_id
        self.enrolled_courses: List['Course'] = []  # 关联
    
    def enroll(self, course: 'Course'):
        self.enrolled_courses.append(course)
        course.students.append(self)

class PhdStudent(Student, Researcher):  # 多重继承
    def __init__(self, name: str, birth_date: date, student_id: str, research_topic: str):
        Student.__init__(self, name, birth_date, student_id)
        self.research_topic = research_topic
        self.advisor: Professor = None  # 关联
    
    def conduct_research(self, topic: str):  # 实现接口
        print(f"Conducting PhD research on {topic}")

class Course:
    def __init__(self, course_code: str, name: str):
        self.course_code = course_code
        self.name = name
        self.teacher: Teacher = None  # 双向关联
        self.students: List[Student] = []  # 双向关联
    
    def add_student(self, student: Student):
        self.students.append(student)
        student.enrolled_courses.append(self)

class Department:
    def __init__(self, name: str):
        self.name = name
        self.faculty: List[Teacher] = []  # 聚合
        self.courses: List[Course] = []  # 聚合
    
    def hire_teacher(self, teacher: Teacher):
        self.faculty.append(teacher)
    
    def add_course(self, course: Course):
        self.courses.append(course)

class EnrollmentSystem:  # 依赖多个类
    def enroll_student(self, student: Student, course: Course):
        if student not in course.students:
            student.enroll(course)
            return True
        return False

综合案例:带抽象接口和静态方法的电商系统

在这里插入图片描述

from abc import ABC, abstractmethod
from datetime import datetime

# 抽象接口:日志服务
class ILogger(ABC):
    @abstractmethod
    def log(self, message: str):
        pass

# 实现接口的类
class ConsoleLogger(ILogger):
    def log(self, message: str):
        print(f"[{datetime.now()}] {message}")

class FileLogger(ILogger):
    def __init__(self, filename: str):
        self.filename = filename
    
    def log(self, message: str):
        with open(self.filename, "a") as file:
            file.write(f"[{datetime.now()}] {message}\n")

# 带静态方法的工具类
class ValidationUtils:
    @staticmethod
    def is_valid_email(email: str) -> bool:
        return "@" in email and "." in email.split("@")[-1]
    
    @staticmethod
    def is_valid_phone(phone: str) -> bool:
        return phone.isdigit() and len(phone) >= 7

# 使用接口和静态方法的类
class UserService:
    def __init__(self, logger: ILogger):
        self.logger = logger
    
    def register_user(self, name: str, email: str, phone: str):
        # 使用静态方法验证
        if not ValidationUtils.is_valid_email(email):
            self.logger.log(f"Invalid email: {email}")
            return False
        
        if not ValidationUtils.is_valid_phone(phone):
            self.logger.log(f"Invalid phone: {phone}")
            return False
        
        # 注册逻辑...
        self.logger.log(f"User {name} registered with {email}")
        return True

# 工厂类(使用静态方法创建对象)
class LoggerFactory:
    @staticmethod
    def create_logger(logger_type: str) -> ILogger:
        if logger_type == "console":
            return ConsoleLogger()
        elif logger_type == "file":
            return FileLogger("app.log")
        else:
            raise ValueError("Invalid logger type")

UML类图要素总结

UML要素Python代码表现UML符号说明
类(Class)class Person:矩形框包含类名、属性和方法
抽象类class AbstractVehicle(ABC):斜体类名包含抽象方法
接口class Interface(ABC): + @abstractmethod<<interface>> + 斜体名称只包含抽象方法
属性self.name: str+name: str+公有, -私有, #保护
方法def get_age(self):
静态方法: @staticmethod 装饰器
类方法:@classmethod 装饰器
抽象方法: @abstractmethod 装饰器
+get_age(): int
静态方法:{static} 标记或方法名下划线
类方法:{classmethod} 标记
抽象方法:{abstract} 标记 + 斜体方法名
类行为定义
继承class Teacher(Person):空心三角+实线泛化关系(is-a)
实现class Professor(Researcher):空心三角+虚线实现接口方法
组合一对一:self.cart = ShoppingCart()
一对多:self.car=[Wheel(),Wheel(),Wheel(),Wheel(),Engine()]
实心菱形+实线强拥有关系(同生命周期)
聚合self.fleet: List[Vehicle] = []空心菱形+实线弱拥有关系(可独立存在)
关联self.teacher: Teacher = None实线箭头对象间持久引用关系
依赖def process_payment(payment):虚线箭头临时使用(方法参数或局部变量中)
枚举class VehicleType(Enum):<<enumeration>>固定值集合
多重继承class Professor(Teacher, Researcher):多个空心三角继承多个父类

一些细节

关于【箭头方向】

箭头方向在UML中表示导航性(Navigability):

箭头类型表示代码等价
无箭头双向导航(默认)双方相互持有引用
单向导航只有源头类知道目标类
双向导航双方相互持有引用
◁/▷箭头端为被引用方箭头指向的类是被持有的类

关于类关系的虚实

以下是对类图中类关系图形的完整总结,重点说明实心/空心、实线/虚线的区别:

1. 实线 vs 虚线
线条类型关系强度生命周期代码对应典型关系
实线强关系可能绑定成员变量(属性)关联、聚合、组合
虚线弱关系无绑定方法参数/局部变量依赖、接口实现
2. 实心 vs 空心
填充类型所有权关系强度典型符号位置代表关系
实心强所有权最强菱形端组合关系
空心弱所有权中等菱形端/三角端聚合/继承/实现
完整关系对比图

在这里插入图片描述

关于【多重性】

多重性定义对象之间的数量关系,常见表示法:

表示法含义示例说明
1恰好1个每个人有1个心脏(组合关系)
0..10或1个学生可能有0或1个导师(关联关系)
1..*1个或多个订单必须包含至少1个商品(组合关系)
0..*0或多个部门可以有0或多个员工(聚合关系)
n恰好n个三角形有3条边(组合关系)
m..nm到n个课程有3-50名学生(关联关系)
*无限多个(同0..*社交媒体用户有多个好友(关联关系)

汇总

要素类型UML表示法代码表现多重性箭头方向生命周期关系
类(Class)矩形框(类名、属性、方法)class MyClass:不适用独立存在
抽象类类名斜体class MyClass(ABC):不适用独立存在
接口<<interface>> + 类框或圆圈class MyInterface(ABC):不适用独立存在
枚举<<enumeration>> + 枚举值class MyEnum(Enum):不适用独立存在
属性[可见性] 属性名: 类型 [= 默认值]self.attr = value不适用随对象存在
方法[可见性] 方法名(参数): 返回类型def method(self):不适用随对象存在
抽象方法斜体或{abstract}@abstractmethod不适用随抽象类存在
静态方法{static} 或下划线@staticmethod不适用类加载时存在
类方法{classmethod}@classmethod不适用类加载时存在
继承(泛化)空心三角箭头 + 实线class Child(Parent):不适用子类→父类子类依赖父类
接口实现空心三角箭头 + 虚线实现接口所有方法不适用实现类→接口实现类依赖接口
关联实线(可带箭头)类属性为另一类对象两端可设置可选(表示导航方向)相互独立
聚合空心菱形 + 实线外部传入对象(self.parts = [ext_obj])整体端通常为1菱形→整体部分可独立于整体
组合实心菱形 + 实线内部创建对象(self.part = Part())整体端通常为1菱形→整体部分依赖整体
依赖虚线箭头局部变量/参数/静态调用不适用使用方→被依赖方临时关系

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

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

相关文章

软件工程的定义与发展历程

文章目录 一、软件工程的定义二、软件工程的发展历程1. 前软件工程时期(1940s-1960s)2. 软件工程诞生(1968)3. 结构化方法时期(1970s)4. 面向对象时期(1980s)5. 现代软件工程(1990s-至今) 三、软件工程的发展趋势 一、软件工程的定义 软件工程是应用系统化、规范化、可量化的方…

第十三节:第五部分:集合框架:集合嵌套

集合嵌套案例分析 代码&#xff1a; package com.itheima.day27_Collection_nesting;import java.util.*;/*目标:理解集合的嵌套。 江苏省 "南京市","扬州市","苏州市","无锡市","常州市" 湖北省 "武汉市","…

Java设计模式之观察者模式详解

一、观察者模式简介 观察者模式&#xff08;Observer Pattern&#xff09;是一种行为型设计模式&#xff0c;它定义了对象之间的一对多依赖关系。当一个对象&#xff08;主题&#xff09;的状态发生改变时&#xff0c;所有依赖于它的对象&#xff08;观察者&#xff09;都会自…

freeRTOS 消息队列之一个事件添加到消息队列超时怎么处理

一 消息队列的结构框图 xTasksWaitingToSend‌&#xff1a;这个列表存储了所有因为队列已满而等待发送消息的任务。当任务尝试向一个已满的队列发送消息时&#xff0c;该任务会被挂起并加入到xTasksWaitingToSend列表中&#xff0c;直到队列中有空间可用‌&#xff0c; xTasksW…

Authpf(OpenBSD)认证防火墙到ssh连接到SSH端口转发技术栈 与渗透网络安全的关联 (RED Team Technique )

目录 &#x1f50d; 1. Authpf概述与Shell设置的作用 什么是Authpf&#xff1f; Shell设置为/usr/sbin/authpf的作用与含义 &#x1f6e0;️ 2. Authpf工作原理与防火墙绕过机制 技术栈 工作原理 防火墙绕过机制 Shell关联 &#x1f310; 3. Authpf与SSH认证及服务探测…

组合与排列

组合与排列主要有两个区别&#xff0c;区别在于是否按次序排列和符号表示不同。 全排列&#xff1a; 从n个不同元素中任取m&#xff08;m≤n&#xff09;个元素&#xff0c;按照一定的顺序排列起来&#xff0c;叫做从n个不同元素中取出m个元素的一个排列。当mn时所有的排列情况…

Apache Druid

目录 Apache Druid是什么&#xff1f; CVE-2021-25646(Apache Druid代码执行漏洞) Apache Druid是什么&#xff1f; Apache Druid是一个高性能、分布式的数据存储和分析系统。设计用于处理大量实时数据&#xff0c;并进行低延迟的查询。它特别适合用于分析大规模日志、事件数据…

使用深蓝词库软件导入自定义的词库到微软拼音输入法

我这有一个人员名单&#xff0c;把它看作一个词库&#xff0c;下面我演示一下如何把这个词库导入微软输入法 首先建一个text文件&#xff0c;一行写一个词条 下载深蓝词库 按照我这个配置&#xff0c;点击转换&#xff0c;然后在桌面微软输入法那右键&#xff0c;选择设置 点…

使用Node.js分片上传大文件到阿里云OSS

阿里云OSS的分片上传&#xff08;Multipart Upload&#xff09;是一种针对大文件优化的上传方式&#xff0c;其核心流程和关键特性如下&#xff1a; 1. ‌核心流程‌ 分片上传分为三个步骤&#xff1a; 初始化任务‌&#xff1a;调用InitiateMultipartUpload接口创建上传任务…

复变函数中的对数函数及其MATLAB演示

复变函数中的对数函数及其MATLAB演示 引言 在实变函数中&#xff0c;对数函数 ln ⁡ x \ln x lnx定义在正实数集上&#xff0c;是一个相对简单的概念。然而&#xff0c;当我们进入复变函数领域时&#xff0c;对数函数展现出更加丰富和复杂的性质。本文将介绍复变函数中对数函…

【Linux】Linux程序地址基础

参考博客&#xff1a;https://blog.csdn.net/sjsjnsjnn/article/details/125533127 一、地址空间的阐述 1.1 程序地址空间 下面的图片展示了程序地址空间的组成结构 我们通过代码来验证一下 int g_unval; int g_val 100;int main(int argc, char *argv[]);void test1() {i…

将图形可视化工具的 Python 脚本打包为 Windows 应用程序

前文我们已经写了一个基于python的tkinter库和matplotlib库的图形可视化工具。 基于Python的tkinter库的图形可视化工具&#xff08;15种图形的完整代码&#xff09;:基于Python的tkinter库的图形可视化工具&#xff08;15种图形的完整代码&#xff09;-CSDN博客 在前文基础上&…

无人机军用与民用技术对比分析

一、材料区别 军用无人机&#xff1a; 1. 高强度特种材料&#xff1a; 大量使用钛合金、碳纤维复合材料&#xff0c;兼顾轻量化与高强度&#xff0c;提升抗冲击性和隐身性能。 关键部件依赖进口材料。 2. 隐身涂层&#xff1a; 采用雷达吸波材料和低红外特征涂料&#xf…

刷leetcode hot100--矩阵6/1

1.螺旋矩阵【很久】6/1【感觉就是思路的搬运工&#xff0c;没完全理解】 54. 螺旋矩阵 - 力扣&#xff08;LeetCode&#xff09; 原来想 但是如果是奇数矩阵&#xff0c;遍历不到中间 解决思路&#xff1a; 用left,right,top,down标记/限定每次遍历的元素&#xff0c;每次从…

Docker轻松搭建Neo4j+APOC环境

Docker轻松搭建Neo4jAPOC环境 一、简介二、Docker部署neo4j三、Docker安装APOC插件四、删除数据库/切换数据库 一、简介 Neo4j 是一款高性能的 原生图数据库&#xff0c;采用 属性图模型 存储数据&#xff0c;支持 Cypher查询语言&#xff0c;适用于复杂关系数据的存储和分析。…

定制开发开源AI智能名片S2B2C商城小程序在无界零售中的应用与行业智能升级示范研究

摘要&#xff1a;本文聚焦无界零售背景下京东从零售产品提供者向零售基础设施提供者的转变&#xff0c;探讨定制开发开源AI智能名片S2B2C商城小程序在这一转变中的应用。通过分析该小程序在商业运营成本降低、效率提升、用户体验优化等方面的作用&#xff0c;以及其与京东AI和冯…

【大模型:知识图谱】--5.neo4j数据库管理(cypher语法2)

目录 1.节点语法 1.1.CREATE--创建节点 1.2.MATCH--查询节点 1.3.RETURN--返回节点 1.4.WHERE--过滤节点 2.关系语法 2.1.创建关系 2.2.查询关系 3.删除语法 3.1.DELETE 删除 3.2.REMOVE 删除 4.功能补充 4.1.SET &#xff08;添加属性&#xff09; 4.2.NULL 值 …

贪心算法应用:装箱问题(BFD算法)详解

贪心算法应用&#xff1a;装箱问题(BFD算法)详解 1. 装箱问题与BFD算法概述 1.1 装箱问题定义 装箱问题(Bin Packing Problem)是组合优化中的经典问题&#xff0c;其定义为&#xff1a; 给定n个物品&#xff0c;每个物品有大小wᵢ (0 < wᵢ ≤ C)无限数量的箱子&#xf…

编程技能:格式化打印05,格式控制符

专栏导航 本节文章分别属于《Win32 学习笔记》和《MFC 学习笔记》两个专栏&#xff0c;故划分为两个专栏导航。读者可以自行选择前往哪个专栏。 &#xff08;一&#xff09;WIn32 专栏导航 上一篇&#xff1a;编程技能&#xff1a;格式化打印04&#xff0c;sprintf 回到目录…

MPLAB X IDE ​软件安装与卸载

1、下载MPLAB X IDE V6.25 MPLAB X IDE | Microchip Technology 正常选Windows&#xff0c;点击Download&#xff0c;等待自动下载完成&#xff1b; MPLAB X IDE 一台电脑上可以安装多个版本&#xff1b; 2、安装MPLAB X IDE V6.25 右键以管理员运行&#xff1b;next; 勾选 I a…