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

从零到一:手把手教你用苍穹外卖项目搞定Spring Boot多表关联(附完整E-R图与避坑指南)

从零到一手把手教你用苍穹外卖项目搞定Spring Boot多表关联附完整E-R图与避坑指南当你第一次面对一个真实的外卖系统开发需求时那些看似简单的分类-菜品-套餐关系背后往往隐藏着令人头疼的数据库设计难题。记得我接手第一个外卖平台项目时就曾被多表关联的复杂逻辑折磨得彻夜难眠——为什么修改套餐要先删除所有关联菜品为什么起售状态要跨三张表校验这些问题在教科书式的CRUD示例中永远找不到答案。本文将带你用Spring Boot和MyBatis从零构建苍穹外卖的核心数据模型我会用真实的E-R图和可落地的代码示例展示如何处理那些只有实战才会遇到的魔鬼细节。不同于市面上泛泛而谈的教程我们将聚焦三个关键问题如何设计可扩展的多对多关系如何用事务保证数据一致性以及那些教科书不会告诉你的业务约束处理技巧。1. 数据库设计从业务需求到E-R图1.1 核心实体关系分析任何外卖系统的数据模型都始于四个基础实体分类(Category)如中式快餐、西式简餐等菜品(Dish)具体的食品项包含价格、描述等属性口味(DishFlavor)如辣度、甜度等可选项套餐(Setmeal)由多个菜品组成的组合它们的关系用一句话概括一个分类包含多个菜品和套餐一个套餐包含多个菜品一个菜品可以有多个口味选项。这种多对多关系需要通过中间表实现CREATE TABLE setmeal_dish ( id BIGINT PRIMARY KEY, setmeal_id BIGINT NOT NULL, dish_id BIGINT NOT NULL, copies INT COMMENT 份数, FOREIGN KEY (setmeal_id) REFERENCES setmeal(id), FOREIGN KEY (dish_id) REFERENCES dish(id) );1.2 完整E-R图解析关键设计要点分离菜品与口味将动态属性抽离到DishFlavor表避免JSON字段的查询瓶颈软删除设计所有表包含status字段控制上下架状态冗余字段优化套餐表中存储分类ID避免多层JOIN提示实际项目中建议为setmeal_dish添加唯一索引UNIQUE(setmeal_id, dish_id)防止重复添加相同菜品2. Spring Boot中的多表操作实战2.1 套餐新增的原子性实现新增套餐需要同时操作setmeal和setmeal_dish表典型的事务应用场景Service RequiredArgsConstructor public class SetmealServiceImpl implements SetmealService { private final SetmealMapper setmealMapper; private final SetmealDishMapper setmealDishMapper; Transactional public void saveWithDishes(SetmealDTO setmealDTO) { // 1. 保存套餐基础信息 Setmeal setmeal new Setmeal(); BeanUtils.copyProperties(setmealDTO, setmeal); setmealMapper.insert(setmeal); // 2. 保存套餐菜品关系 ListSetmealDish setmealDishes setmealDTO.getSetmealDishes() .stream() .map(dish - { dish.setSetmealId(setmeal.getId()); return dish; }).collect(Collectors.toList()); setmealDishMapper.insertBatch(setmealDishes); } }常见坑点忘记设置setmeal_dish中的setmeal_id关联批量插入时未考虑数据库方言差异事务未生效导致部分失败检查Transactional是否被自调用绕过2.2 级联删除的正确姿势删除套餐时需要同步清理关联菜品关系但必须遵守业务规则Transactional public void removeWithDishes(ListLong ids) { // 1. 检查是否可删除起售状态 if (setmealMapper.countStatusByIds(ids, StatusEnum.ENABLE) 0) { throw new BusinessException(存在启售套餐无法删除); } // 2. 删除套餐记录 setmealMapper.deleteBatchIds(ids); // 3. 删除关联菜品 setmealDishMapper.deleteBySetmealIds(ids); }这里使用deleteBySetmealIds而非循环单条删除性能提升5-8倍基于JMH测试3. 复杂查询的MyBatis优化技巧3.1 分页查询与联表优化套餐列表需要显示分类名称典型的JOIN应用场景select idpageQuery resultTypecom.sky.vo.SetmealVO SELECT s.*, c.name AS categoryName FROM setmeal s LEFT JOIN category c ON s.category_id c.id where if testname ! null AND s.name LIKE CONCAT(%,#{name},%) /if if testcategoryId ! null AND s.category_id #{categoryId} /if if teststatus ! null AND s.status #{status} /if /where ORDER BY s.update_time DESC /select性能对比查询方式平均耗时(ms)适用场景单表多次查询120简单关联LEFT JOIN45中等数据量冗余字段22高频访问字段3.2 状态一致性校验起售套餐时需要确保所有关联菜品已启售public void startOrStop(Integer status, Long id) { if (status StatusEnum.ENABLE) { // 检查套餐包含停售菜品 ListDish disabledDishes dishMapper.listDisabledBySetmealId(id); if (!disabledDishes.isEmpty()) { throw new BusinessException(套餐包含未启售菜品 disabledDishes.stream() .map(Dish::getName) .collect(Collectors.joining(,))); } } Setmeal setmeal Setmeal.builder() .id(id) .status(status) .build(); setmealMapper.updateById(setmeal); }这个案例展示了业务规则如何影响技术实现——没有银弹解决方案必须根据实际约束编写校验逻辑。4. 高频问题解决方案4.1 先删后增的更新模式为什么修改套餐要先删除所有关联菜品再重新添加原因有三避免局部更新前端可能只提交部分菜品变更简化逻辑统一视为全新关系比差异对比更可靠历史记录某些场景需要完整变更轨迹实现示例Transactional public void updateWithDishes(SetmealDTO setmealDTO) { // 1. 更新基础信息 Setmeal setmeal new Setmeal(); BeanUtils.copyProperties(setmealDTO, setmeal); setmealMapper.updateById(setmeal); // 2. 清理旧关联 setmealDishMapper.deleteBySetmealId(setmeal.getId()); // 3. 插入新关联 ListSetmealDish setmealDishes setmealDTO.getSetmealDishes(); if (!CollectionUtils.isEmpty(setmealDishes)) { setmealDishes.forEach(dish - dish.setSetmealId(setmeal.getId())); setmealDishMapper.insertBatch(setmealDishes); } }4.2 缓存与事务的协同问题当引入Redis缓存时要注意事务提交与缓存更新的顺序Transactional public void updateStatus(Integer status, Long id) { // 数据库操作 Setmeal setmeal Setmeal.builder() .id(id) .status(status) .build(); setmealMapper.updateById(setmeal); // 缓存操作在事务提交后执行 TransactionSynchronizationManager.registerSynchronization( new TransactionSynchronization() { Override public void afterCommit() { redisTemplate.delete(setmeal: id); } }); }这种模式避免了脏缓存问题确保缓存始终与数据库一致。

相关文章:

从零到一:手把手教你用苍穹外卖项目搞定Spring Boot多表关联(附完整E-R图与避坑指南)

从零到一:手把手教你用苍穹外卖项目搞定Spring Boot多表关联(附完整E-R图与避坑指南) 当你第一次面对一个真实的外卖系统开发需求时,那些看似简单的"分类-菜品-套餐"关系背后,往往隐藏着令人头疼的数据库设计…...

超维USV-M1000无人船实战:手把手教你搞定元生RTK、元厚测深仪与QGC地面站联调

超维USV-M1000无人船多设备联调实战:从RTK配置到QGC地面站深度整合 当超维USV-M1000无人船从包装箱中取出时,摆在工程师面前的从来不是一台即插即用的设备,而是一个需要精密调校的测绘系统集成平台。这款专为水域测绘设计的无人船&#xff0c…...

STM32光敏传感器实战:从硬件连接到智能路灯控制(附完整代码)

STM32光敏传感器实战:从硬件连接到智能路灯控制(附完整代码) 在物联网和智能家居快速发展的今天,环境光检测已成为许多自动化系统的基础功能。作为一名嵌入式开发者,掌握光敏传感器与STM32微控制器的集成应用&#xff…...

2025最权威的五大降AI率方案实测分析

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 降低AI生成内容的可识别性之为,要从语言风格开始着手,还要从逻辑结构…...

别再用鼠标拖滚动条了!GoLand 2022.2.3 这几个插件让你的代码阅读效率翻倍

GoLand 2022.2.3 插件组合:打造专业开发者的代码阅读工作流 作为一名长期与Go代码打交道的开发者,你是否经历过这样的场景:接手一个数万行代码的项目时,面对密密麻麻的函数和结构体,像在迷宫中寻找出口;或是…...

2026届毕业生推荐的五大降重复率方案推荐榜单

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 为能切实且有效地把文本里AIGC的比例给降下来,特地给出建议采用多维度的策略&am…...

Bedrock Launcher全攻略:解决Minecraft基岩版多版本管理难题的创新方案

Bedrock Launcher全攻略:解决Minecraft基岩版多版本管理难题的创新方案 【免费下载链接】BedrockLauncher 项目地址: https://gitcode.com/gh_mirrors/be/BedrockLauncher Minecraft基岩版玩家常面临版本管理的困境:想体验新特性需放弃现有存档&…...

保姆级教程:为嵌入式Linux(ARM/AArch64)交叉编译带完整符号支持的Perf工具

ARM架构嵌入式Linux系统性能调优实战:Perf工具深度定制指南 在嵌入式系统开发中,性能优化往往是最具挑战性的环节之一。当你的应用在ARM或AArch64架构的嵌入式设备上运行时出现卡顿、延迟或资源耗尽,传统的打印调试和日志分析往往难以定位深…...

新手入门:利用快马平台快速理解与模拟openclaw重启命令执行

今天想和大家分享一个特别适合机器人编程新手的实践项目——如何在安全可控的环境下理解和模拟openclaw机械爪的重启命令。这个过程中我发现了InsCode(快马)平台这个神器,它让整个学习过程变得异常简单。 为什么需要专门学习重启命令? 机械爪作为机器人常…...

用快马平台5分钟打造heic转jpg在线工具原型,零代码实现图片格式转换

最近工作中遇到一个需求:需要把手机拍摄的HEIC格式照片批量转换成JPG格式。这种格式转换的需求其实很常见,特别是现在iPhone默认都使用HEIC格式保存照片。作为一个前端开发者,我决定用InsCode(快马)平台快速实现一个在线转换工具的原型。 需求…...

利用快马平台为dhnvr416h-hd设备快速构建交互式原型模拟器

最近在做一个智能硬件项目,需要为dhnvr416h-hd设备开发一个快速原型模拟器。这个模拟器主要用于验证设备接口和功能逻辑,避免直接操作真实设备带来的风险。经过一番摸索,我发现用InsCode(快马)平台可以非常高效地完成这个任务,下面…...

相场模拟——合金,金属凝固模型,各向异性枝晶生长karma 合金凝固模型,选区激光熔融,激光增...

相场模拟——合金,金属凝固模型,各向异性枝晶生长karma 合金凝固模型,选区激光熔融,激光增材制造,选择性激光熔融,SLM,定向凝固,熔铸 1matlab,实现合金各向异性枝晶生长&…...

二维码逆向工程:从01二进制到可扫描二维码的完整流程

二维码逆向工程:从01二进制到可扫描二维码的完整流程 二维码已成为现代生活中不可或缺的信息载体,但你是否想过,一串简单的0和1如何转化为可扫描的二维码?本文将带你深入探索二维码的逆向工程世界,从二进制数据处理到图…...

收放系统变深声纳(VDS)

一、整体结构与典型 VDS 收放系统框图 从 Curtiss-Wright 的公开资料看,典型舰用 VDS 收放系统通常包括: 水动力稳定、声学透明的拖体(4.5–18英尺 / 1.37–5.49 m,钢制或复合材料,360 水平声窗) 拖体发射/回收机构(Body launch and recovery mechanism) 拖缆绞车(拖…...

实战踩坑:用Dify+DeepSeek对接MySQL,我遇到的5个典型错误和解决方案

实战踩坑:用DifyDeepSeek对接MySQL,我遇到的5个典型错误和解决方案 当Dify工作流遇上DeepSeek模型,再结合MySQL数据库查询,这个技术组合听起来很美好,但实际操作中却暗藏不少"坑"。作为已经踩过这些坑的开发…...

大量频繁发起连接导致瀚高数据库进程被信号6杀死

文章目录环境症状问题原因解决方案环境 系统平台:银河麒麟 (龙芯) 版本:4.5.1 症状 数据库版本:瀚高安全版V4.5.1.2 介质 hgdb-see-4.5.1.2-ee53424.loongarch64.rpm 信号6的报错信息: 2025-04-18 08…...

如何用GHelper替代Armoury Crate,让华硕笔记本性能与续航双丰收?

如何用GHelper替代Armoury Crate,让华硕笔记本性能与续航双丰收? 【免费下载链接】g-helper Lightweight, open-source control tool for ASUS laptops and ROG Ally. Manage performance modes, fans, GPU, battery, and RGB lighting across Zephyrus,…...

八大网盘直链下载助手:免费获取高速下载链接的完整指南

八大网盘直链下载助手:免费获取高速下载链接的完整指南 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 ,支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼…...

毕业工具怎么选?一张表对比选题、写作、降重、排版、答辩核心功能

从选题到答辩,选对工具能省下80%的无效劳动。本文聚焦毕业全流程各环节的核心功能,帮你快速定位适合自己的工具组合。一、毕业全流程功能总览一个完整的毕业论文写作流程通常包括以下几个阶段,每个阶段都有对应的工具功能可以提效&#xff1a…...

本科生毕业论文必备:2026年这几款工具从开题到答辩全覆盖

基于2026年最新实测数据,从学术严谨性、降重效果、AIGC合规性等维度,为你筛选出综合实力最强的论文写作工具。 一、综合实力TOP 10榜单 排名工具名称核心定位综合评分价格一句话总结🥇1毕业之家全流程学术管家9.4/101.2元/千字起真实文献三级…...

Anaconda3 2025 面向数据科学安装教程:详细步骤+自定义路径+Navigator启动)

其包含了conda、Python等180多个科学包及其依赖项。Anaconda可以看做Python的一个集成安装,它不仅免去了许多复杂的环境搭建,还内置了许多使用的Python工具 一、安装准备 安装包下载:https://pan.xunlei.com/s/VOpVUmfa4taHwZ-gAYIVqvCuA1?…...

利用快马平台ai快速原型开发,十分钟搭建instagram内容下载器demo

最近想验证一个Instagram内容下载器的功能可行性,但自己从头开发太耗时。尝试用InsCode(快马)平台的AI辅助功能,没想到十分钟就搭出了可运行的Demo原型。记录下这个快速验证过程,或许对需要快速原型开发的朋友有参考价值。 明确核心需求 首先…...

探索TMSpeech:解锁Windows本地实时语音转文字的高效工作流

探索TMSpeech:解锁Windows本地实时语音转文字的高效工作流 【免费下载链接】TMSpeech 腾讯会议摸鱼工具 项目地址: https://gitcode.com/gh_mirrors/tm/TMSpeech 还在为会议记录手忙脚乱?还在为在线课程跟不上节奏而烦恼?TMSpeech为你…...

【Mojo与Python混合编程高阶实战】:20年专家亲授5大避坑指南与性能翻倍技巧

第一章:Mojo与Python混合编程的底层原理与生态定位Mojo 是一种专为 AI 系统编程设计的现代系统语言,其核心目标是在保留 Python 语法亲和力的同时,提供接近 C/Rust 的性能与底层控制能力。Mojo 与 Python 并非简单互调关系,而是通…...

WarcraftHelper:魔兽争霸III现代优化解决方案全玩家实战指南

WarcraftHelper:魔兽争霸III现代优化解决方案全玩家实战指南 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper WarcraftHelper是一款专为魔兽…...

仅限TOP20量化机构内部流通的C++内存池测试矩阵(含FPGA协处理器适配层验证项),现在限时解密!

第一章:金融级C内存池测试的范式演进与行业背景在高频交易、实时风控与低延迟结算等金融核心系统中,内存分配性能直接决定毫秒级响应能力的上限。传统 malloc/free 在高并发小对象场景下易引发锁争用、碎片化加剧与缓存行失效,迫使头部券商与…...

UE6.5调试性能对比实测:Clang 19 vs MSVC 17.12 vs GCC 14.2(C++27特性支持度+调试信息完整性双维度TOP1)

第一章:UE6.5 C27 调试能力演进与基准定位Unreal Engine 6.5 首次原生支持 C27 标准子集,并深度整合了 Clang 18 的调试元数据增强特性,显著提升了符号解析精度与运行时诊断能力。相比 UE5.4 中基于 DWARF-5 的有限 C20 支持,UE6.…...

UE5游戏逆向实战:用FModel提取.pak文件中的3D模型(附Dumper-7避坑指南)

UE5游戏逆向实战:用FModel提取.pak文件中的3D模型(附Dumper-7避坑指南) 在虚幻引擎5(UE5)游戏开发与逆向工程领域,资源提取始终是开发者与爱好者关注的核心技能。随着引擎版本迭代,传统的.pak文…...

如何解决Jellyfin番剧管理痛点?Bangumi插件的技术实现与应用指南

如何解决Jellyfin番剧管理痛点?Bangumi插件的技术实现与应用指南 【免费下载链接】jellyfin-plugin-bangumi bgm.tv plugin for jellyfin 项目地址: https://gitcode.com/gh_mirrors/je/jellyfin-plugin-bangumi Jellyfin作为开源媒体中心软件,在…...

颠覆认知:重新定义CPU性能边界的智能优化指南

颠覆认知:重新定义CPU性能边界的智能优化指南 【免费下载链接】CPUDoc 项目地址: https://gitcode.com/gh_mirrors/cp/CPUDoc 当我们谈论电脑性能时,大多数人会想到升级硬件或超频,但真正的性能瓶颈往往藏在系统调度的细节里。本文将…...