当前位置: 首页 > news >正文

深入理解系统:UML类图

UML类图

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

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

案例1:电商系统类图

在这里插入图片描述

# UML类图元素:类、属性、方法、继承、关联、聚合
from typing import Listclass 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.__passwordclass 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_idself.name = nameself.price = priceclass 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 = itemsself.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_idself.name = nameself.products: List[Product] = []  # 聚合关系# 接口实现 (依赖倒置)
class PaymentProcessor(ABC):  # 抽象类/接口@abstractmethoddef process(self, amount: float):passclass 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 dateclass VehicleType(Enum):  # 枚举类CAR = 1TRUCK = 2SUV = 3MOTORCYCLE = 4class AbstractVehicle(ABC):  # 抽象类def __init__(self, license_plate: str, model: str, year: int):self.license_plate = license_plateself.model = modelself.year = yearself.available = True@abstractmethoddef get_rental_rate(self) -> float:passclass Car(AbstractVehicle):  # 继承def __init__(self, license_plate: str, model: str, year: int, seats: int):super().__init__(license_plate, model, year)self.seats = seatsdef 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 = nameself.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 = Falsereturn contractreturn Noneclass Customer:def __init__(self, customer_id: str, name: str):self.customer_id = customer_idself.name = nameself.license_number = ""class RentalContract:  # 组合类def __init__(self, customer: Customer, vehicle: AbstractVehicle, start_date: date, end_date: date):self.customer = customerself.vehicle = vehicleself.start_date = start_dateself.end_date = end_dateself.total_cost = self.calculate_cost()def calculate_cost(self) -> float:days = (self.end_date - self.start_date).daysreturn 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 dateclass Person:def __init__(self, name: str, birth_date: date):self.name = nameself.birth_date = birth_datedef get_age(self) -> int:today = date.today()return today.year - self.birth_date.yearclass Researcher(ABC):  # 接口@abstractmethoddef conduct_research(self, topic: str):passclass Teacher(Person):  # 单继承def __init__(self, name: str, birth_date: date, department: str):super().__init__(name, birth_date)self.department = departmentself.courses: List[Course] = []  # 双向关联def assign_course(self, course: 'Course'):self.courses.append(course)course.teacher = selfclass Professor(Teacher, Researcher):  # 多重继承def __init__(self, name: str, birth_date: date, department: str, title: str):Teacher.__init__(self, name, birth_date, department)self.title = titledef conduct_research(self, topic: str):  # 实现接口print(f"Conducting research on {topic}")def supervise_phd(self, student: 'PhdStudent'):student.advisor = selfclass Student(Person):def __init__(self, name: str, birth_date: date, student_id: str):super().__init__(name, birth_date)self.student_id = student_idself.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_topicself.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_codeself.name = nameself.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 = nameself.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 Truereturn False

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

在这里插入图片描述

from abc import ABC, abstractmethod
from datetime import datetime# 抽象接口:日志服务
class ILogger(ABC):@abstractmethoddef 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 = filenamedef log(self, message: str):with open(self.filename, "a") as file:file.write(f"[{datetime.now()}] {message}\n")# 带静态方法的工具类
class ValidationUtils:@staticmethoddef is_valid_email(email: str) -> bool:return "@" in email and "." in email.split("@")[-1]@staticmethoddef is_valid_phone(phone: str) -> bool:return phone.isdigit() and len(phone) >= 7# 使用接口和静态方法的类
class UserService:def __init__(self, logger: ILogger):self.logger = loggerdef register_user(self, name: str, email: str, phone: str):# 使用静态方法验证if not ValidationUtils.is_valid_email(email):self.logger.log(f"Invalid email: {email}")return Falseif 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:@staticmethoddef 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 Researcher(ABC):<<interface>>只包含抽象方法
属性self.name: str+name: str+公有, -私有, #保护
方法def get_age(self):+get_age(): int类行为定义
继承class Teacher(Person):空心三角+实线泛化关系(is-a)
实现class Professor(Researcher):空心三角+虚线实现接口方法
组合self.cart = ShoppingCart()实心菱形+实线强拥有关系(同生命周期)
聚合self.fleet: List[Vehicle] = []空心菱形+实线弱拥有关系(可独立存在)
关联self.teacher: Teacher = None实线箭头对象间持久引用关系
依赖def process_payment(payment):虚线箭头临时使用关系
枚举class VehicleType(Enum):<<enumeration>>固定值集合
多重继承class Professor(Teacher, Researcher):多个空心三角继承多个父类
UML元素UML表示法Python实现
抽象接口<<interface>> + 斜体名称class Interface(ABC): + @abstractmethod
接口实现虚线空心三角箭头 (…|>)实现接口的所有抽象方法
静态方法{static} 标记或方法名下划线@staticmethod 装饰器
类方法{classmethod} 标记@classmethod 装饰器
抽象方法{abstract} 标记 + 斜体方法名@abstractmethod 装饰器
依赖关系虚线箭头 (–>)方法参数或局部变量中使用
创建关系虚线箭头 + <<create>> 构造型工厂方法创建对象

一些细节

关于【箭头方向】

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

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

关于【多重性】

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

表示法含义示例说明
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菱形→整体部分依赖整体
依赖虚线箭头局部变量/参数/静态调用不适用使用方→被依赖方临时关系

相关文章:

  • Java面试专项一-准备篇
  • STM32实战:智能环境监测站设计方案
  • URL 结构说明+路由(接口)的认识
  • Cisco IOS XE WLC 任意文件上传漏洞复现(CVE-2025-20188)
  • 【25软考网工】第十章 网络规划与设计(1)综合布线
  • Bresenham算法
  • 打造可扩展的大模型训练框架:支持PEFT微调、分布式训练与TensorBoard可视化
  • go语言学习 第5章:函数
  • 如何选择合适的embedding模型用于非英文语料
  • 【PmHub面试篇】PmHub 整合 TransmittableThreadLocal(TTL)缓存用户数据面试专题解析
  • 基于Gemini 2.5 Pro打造的AI智能体CanvasX上线,绘制常见图表(折线图、柱状图等),国内直接使用
  • [Java 基础]对象,膜具倒出来的
  • 微信小程序实现运动能耗计算
  • 12306高并发计算架构揭秘:Apache Geode 客户端接入与实践
  • webPack基本使用步骤
  • Neo4j 监控全解析:原理、技术、技巧与最佳实践
  • 【Linux系列】rsync命令详解与实践
  • 深入理解C#中的Web API:构建现代化HTTP服务的完整指南
  • BERT:让AI真正“读懂”语言的革命
  • Vue指令修饰符、v-bind对样式控制的增强、computed计算属性、watch监视器
  • 网站开发文档撰写模板/企业网站设计思路
  • wordpress漏洞总结/搜狗网站seo
  • wordpress小说插件/seo项目优化案例分析文档
  • web前端做一个网页/seo优化教程
  • 中建八局第一建设有限公司税号/搜索引擎营销seo
  • 温州做网站建设哪家好/seo排名规则