苍穹外卖 - 菜品起售/停售复习笔记

1. 业务逻辑核心

这个功能不仅仅是"改个状态"那么简单,它包含了一个隐形的级联逻辑

  • 起售 (Status = 1)

  • 逻辑 :非常简单,只修改菜品表 (dish) 的状态即可。

  • 影响:不影响其他表。

  • 停售 (Status = 0)

  • 逻辑"连坐机制"

  • 核心规则:如果一道菜(如"宫保鸡丁")不卖了,那么包含这道菜的所有套餐(如"超值双人餐")也必须立刻停止售卖。

  • 原因:防止用户点了套餐,结果厨房发现缺菜做不出来,导致客诉。


2. 接口设计与参数传递

  • 接口地址POST /admin/dish/status/{status}?id={id}
  • 参数解剖
参数位置 形式 注解 含义
路径 (Path) /status/{status} @PathVariable 做什么操作? (1=起售, 0=停售)
查询 (Query) ?id=xxx Long id 操作哪个对象? (菜品ID)

3. 核心代码实现 (Code Implementation)

3.1 Service 层 (DishServiceImpl)

使用了 Builder模式 构建对象,并使用了 批量更新 优化了级联逻辑。

java 复制代码
    @Override
    @Transactional // 1. 事务控制:保证"菜品停售"和"套餐停售"原子性
    public void startOrStop(Integer status, Long id) {
        // 修改菜品状态
        Dish dish = Dish.builder()
                .id(id)
                .status(status)
                .build();
        dishMapper.update(dish);

        // 如果是停售操作,还需要将关联的套餐也停售
        if (status.equals(StatusConstant.DISABLE)) {
            // 查询关联的套餐id
            List<Long> setmealIds = setmealDishMapper.getSetmealIdsByDishIds(Arrays.asList(id));
            
            // 2. 判空防止 NPE (NullPointerException)
            if (setmealIds != null && !setmealIds.isEmpty()) {
                // 3. 批量将关联的套餐全部停售 (性能优化点)
                setmealMapper.updateStatusBatch(setmealIds, StatusConstant.DISABLE);
            }
        }
    }

3.2 Mapper 层 (动态 SQL)

这里有两个关键的 XML 配置,分别对应单条更新批量更新

(1) DishMapper.xml - 单条动态更新

必须使用 <if> 标签,防止将未传入的字段误更新为 null。

xml 复制代码
    <update id="update">
        update dish
        <set>
            <if test="name != null">name = #{name},</if>
            <if test="categoryId != null">category_id = #{categoryId},</if>
            <if test="price != null">price = #{price},</if>
            <if test="image != null">image = #{image},</if>
            <if test="description != null">description = #{description},</if>
            <if test="status != null">status = #{status},</if>
            <if test="updateTime != null">update_time = #{updateTime},</if>
            <if test="updateUser != null">update_user = #{updateUser},</if>
        </set>
        where id = #{id}
    </update>
(2) SetmealMapper.xml - 批量更新

使用 <foreach> 标签配合 IN 语句,实现一次交互更新多条数据。

xml 复制代码
    <update id="updateStatusBatch">
        update setmeal set status = #{status} where id in
        <foreach collection="setmealIds" item="setmealId" separator="," open="(" close=")">
            #{setmealId}
        </foreach>
    </update>

相关推荐
在荒野的梦想17 小时前
LangChain4j 集成若依单体应用 | 5 大 AI 功能实战:多轮对话、流式输出、RAG 知识库
java·人工智能
苏渡苇17 小时前
ConcurrentHashMap.computeIfAbsent():高并发下安全初始化的终极方案
java·安全·jdk·高并发·hashmap·concurrent
如雨随行202017 小时前
【Vim】学习笔记(9)命令模式
笔记·学习·vim
FinelyYang17 小时前
nginx的docker镜像封禁地区IP
java·nginx·docker
如雨随行202017 小时前
【Vim】学习笔记(8)tips-2
笔记·学习·vim
暴躁小师兄数据学院17 小时前
【WEB3.0零基础转换笔记】Rust编程篇-第4讲:控制流
开发语言·笔记·rust·web3·区块链·智能合约
空空潍17 小时前
Spring AI 实战系列(六):Tool Calling深度实战,让大模型自动调用你的业务接口
java·人工智能·spring
NoSi EFUL17 小时前
Redis6.2.6下载和安装
java
Oll Correct17 小时前
实验十一:地址解析协议ARP不能跨网络直接使用
网络·笔记
老虎062717 小时前
LeetCode热题100 刷题笔记(第五天)多维动态规划(中心扩展法) 「 最长回文子串」
笔记·leetcode·动态规划