# 苍穹外卖 - day07# 课程内容功能实现:缓存商品 、购物车
效果图:
# 缓存菜品# 问题说明用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大。
结果:
系统响应慢、用户体验差
# 实现思路通过 Redis 来缓存菜品数据,减少数据库查询操作。
缓存逻辑分析:
每个分类下的菜品保存一份缓存数据 数据库中菜品数据有变更时清理缓存数据
# 代码开发修改用户端接口 DishController 的 list 方法,加入缓存处理逻辑:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 @Autowired private RedisTemplate redisTemplate; @GetMapping("/list") @ApiOperation("根据分类id查询菜品") public Result<List<DishVO>> list (Long categoryId) { String key = "dish_" + categoryId; List<DishVO> list = (List<DishVO>) redisTemplate.opsForValue().get(key); if (list != null && list.size() > 0 ){ return Result.success(list); } Dish dish = new Dish (); dish.setCategoryId(categoryId); dish.setStatus(StatusConstant.ENABLE); list = dishService.listWithFlavor(dish); redisTemplate.opsForValue().set(key, list); return Result.success(list); }
为了保证数据库 和 Redis 中的数据保持一致,修改管理端接口 DishController 的相关方法,加入清理缓存逻辑。
需要改造的方法:
抽取清理缓存的方法:
在管理端 DishController 中添加
1 2 3 4 5 6 7 8 9 10 @Autowired private RedisTemplate redisTemplate; private void cleanCache (String pattern) { Set keys = redisTemplate.keys(pattern); redisTemplate.delete(keys); }
调用清理缓存的方法,保证数据一致性:
1). 新增菜品优化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @PostMapping @ApiOperation("新增菜品") public Result save (@RequestBody DishDTO dishDTO) { log.info("新增菜品:{}" , dishDTO); dishService.saveWithFlavor(dishDTO); String key = "dish_" + dishDTO.getCategoryId(); cleanCache(key); return Result.success(); }
2). 菜品批量删除优化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @DeleteMapping @ApiOperation("菜品批量删除") public Result delete (@RequestParam List<Long> ids) { log.info("菜品批量删除:{}" , ids); dishService.deleteBatch(ids); cleanCache("dish_*" ); return Result.success(); }
3). 修改菜品优化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @PutMapping @ApiOperation("修改菜品") public Result update (@RequestBody DishDTO dishDTO) { log.info("修改菜品:{}" , dishDTO); dishService.updateWithFlavor(dishDTO); cleanCache("dish_*" ); return Result.success(); }
4). 菜品起售停售优化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @PostMapping("/status/{status}") @ApiOperation("菜品起售停售") public Result<String> startOrStop (@PathVariable Integer status, Long id) { dishService.startOrStop(status, id); cleanCache("dish_*" ); return Result.success(); }
# 功能测试可以通过如下方式进行测试:
查看控制台 sql 前后端联调 查看 Redis 中的缓存数据 以加入缓存 、菜品修改 两个功能测试为例,通过前后端联调方式,查看控制台 sql 的打印和 Redis 中的缓存数据变化。
1). 加入缓存
当第一次查询某个分类的菜品时,会从数据为中进行查询,同时将查询的结果存储到 Redis 中,在后绪的访问,若查询相同分类的菜品时,直接从 Redis 缓存中查询,不再查询数据库。
登录小程序:
选择蜀味牛蛙 (id=17)
查看控制台 sql:
有查询语句,说明是从数据库中进行查询
查看 Redis 中的缓存数据:
说明缓存成功
再次访问:
选择蜀味牛蛙 (id=17)
说明是从 Redis 中查询的数据。
2). 菜品修改
当在后台修改菜品数据时,为了保证 Redis 缓存中的数据和数据库中的数据时刻保持一致,当修改后,需要清空对应的缓存数据。用户再次访问时,还是先从数据库中查询,同时再把查询的结果存储到 Redis 中,这样,就能保证缓存和数据库的数据保持一致。
进入后台:
修改蜀味牛蛙分类下的任意一个菜品,当前分类的菜品数据已在 Redis 中缓存
修改:
查看 Redis 中的缓存数据:
说明修改时,已清空缓存
用户再次访问同一个菜品分类时,需要先查询数据库,再把结果同步到 Redis 中,保证了两者数据一致性。
其它功能测试步骤基本一致,自已测试即可。
# 缓存套餐# Spring Cache# 介绍Spring Cache 是一个框架,实现了基于注解的缓存功能,只需要简单地加一个注解,就能实现缓存功能。
Spring Cache 提供了一层抽象,底层可以切换不同的缓存实现,例如:
EHCache Caffeine Redis (常用) 起步依赖:
1 2 3 4 <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-cache</artifactId > <version > 2.7.3</version > </dependency >
# 常用注解在 SpringCache 中提供了很多缓存操作的注解,常见的是以下的几个:
注解 说明 @EnableCaching 开启缓存注解功能,通常加在启动类上 @Cacheable 在方法执行前先查询缓存中是否有数据,如果有数据,则直接返回缓存数据;如果没有缓存数据,调用方法并将方法返回值放到缓存中 @CachePut 将方法的返回值放到缓存中 @CacheEvict 将一条或多条数据从缓存中删除
在 spring boot 项目中,使用缓存技术只需在项目中导入相关缓存技术的依赖包,并在启动类上使用 @EnableCaching 开启缓存支持即可。
例如,使用 Redis 作为缓存技术,只需要导入 Spring data Redis 的 maven 坐标即可。
# 入门案例1). 环境准备
导入基础工程:
底层已使用 Redis 缓存实现
基础环境的代码,在我们今天的资料中已经准备好了, 大家只需要将这个工程导入进来就可以了。导入进来的工程结构如下:
数据库准备:
创建名为 spring_cache_demo 数据库,将 springcachedemo.sql 脚本直接导入数据库中。
引导类上加 @EnableCaching:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 package com.itheima;import lombok.extern.slf4j.Slf4j;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;import org.springframework.cache.annotation.EnableCaching;@Slf4j @SpringBootApplication @EnableCaching public class CacheDemoApplication { public static void main (String[] args) { SpringApplication.run(CacheDemoApplication.class,args); log.info("项目启动成功..." ); } }
2). @CachePut 注解
@CachePut 说明:
作用:将方法返回值,放入缓存
value: 缓存的名称,每个缓存名称下面可以有很多 key
key: 缓存的 key ----------> 支持 Spring 的表达式语言 SPEL 语法
在 save 方法上加注解 @CachePut
当前 UserController 的 save 方法是用来保存用户信息的,我们希望在该用户信息保存到数据库的同时,也往缓存中缓存一份数据,我们可以在 save 方法上加上注解 @CachePut,用法如下:
1 2 3 4 5 6 7 8 9 10 11 @PostMapping @CachePut(value = "userCache", key = "#user.id") public User save (@RequestBody User user) { userMapper.insert(user); return user; }
说明:
key 的写法如下
#user.id : #user 指的是方法形参的名称,id 指的是 user 的 id 属性,也就是使用 user 的 id 属性作为 key ;
#result.id : #result 代表方法返回值,该表达式 代表以返回对象的 id 属性作为 key ;
#p0.id :#p0 指的是方法中的第一个参数,id 指的是第一个参数的 id 属性,也就是使用第一个参数的 id 属性作为 key ;
#a0.id :#a0 指的是方法中的第一个参数,id 指的是第一个参数的 id 属性,也就是使用第一个参数的 id 属性作为 key ;
#root.args [0].id:#root.args [0] 指的是方法中的第一个参数,id 指的是第一个参数的 id 属性,也就是使用第一个参数
的 id 属性作为 key ;
启动服务,通过 swagger 接口文档测试,访问 UserController 的 save () 方法
因为 id 是自增,所以不需要设置 id 属性
查看 user 表中的数据
查看 Redis 中的数据
3). @Cacheable 注解
@Cacheable 说明:
作用:在方法执行前,spring 先查看缓存中是否有数据,如果有数据,则直接返回缓存数据;若没有数据,调用方法并将方法返回值放到缓存中
value: 缓存的名称,每个缓存名称下面可以有多个 key
key: 缓存的 key ----------> 支持 Spring 的表达式语言 SPEL 语法
在 getById 上加注解 @Cacheable
1 2 3 4 5 6 7 8 9 10 11 @GetMapping @Cacheable(cacheNames = "userCache",key="#id") public User getById (Long id) { User user = userMapper.getById(id); return user; }
重启服务,通过 swagger 接口文档测试,访问 UserController 的 getById () 方法
第一次访问,会请求我们 controller 的方法,查询数据库。后面再查询相同的 id,就直接从 Redis 中查询数据,不用再查询数据库了,就说明缓存生效了。
提前在 redis 中手动删除掉 id=1 的用户数据
查看控制台 sql 语句:
说明从数据库查询的用户数据
查看 Redis 中的缓存数据:
说明已成功缓存
再次查询相同 id 的数据时,直接从 redis 中直接获取,不再查询数据库。
4). @CacheEvict 注解
@CacheEvict 说明:
作用:清理指定缓存
value: 缓存的名称,每个缓存名称下面可以有多个 key
key: 缓存的 key ----------> 支持 Spring 的表达式语言 SPEL 语法
在 delete 方法上加注解 @CacheEvict
1 2 3 4 5 6 7 8 9 10 11 @DeleteMapping @CacheEvict(cacheNames = "userCache",key = "#id") public void deleteById (Long id) { userMapper.deleteById(id); } @DeleteMapping("/delAll") @CacheEvict(cacheNames = "userCache",allEntries = true) public void deleteAll () { userMapper.deleteAll(); }
重启服务,通过 swagger 接口文档测试,访问 UserController 的 deleteAll () 方法
查看 user 表:
数据清空
查询 Redis 缓存数据
# 实现思路实现步骤:
1). 导入 Spring Cache 和 Redis 相关 maven 坐标
2). 在启动类上加入 @EnableCaching 注解,开启缓存注解功能
3). 在用户端接口 SetmealController 的 list 方法上加入 @Cacheable 注解
4). 在管理端接口 SetmealController 的 save、delete、update、startOrStop 等方法上加入 CacheEvict 注解
# 代码开发按照上述实现步骤:
1). 导入 Spring Cache 和 Redis 相关 maven 坐标 (已实现)
1 2 3 4 5 6 7 8 9 <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-data-redis</artifactId > </dependency > <dependency > <groupId > org.springframework.boot</groupId > <artifactId > spring-boot-starter-cache</artifactId > </dependency >
2). 在启动类上加入 @EnableCaching 注解,开启缓存注解功能
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 package com.sky;import lombok.extern.slf4j.Slf4j;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;import org.springframework.cache.annotation.EnableCaching;import org.springframework.transaction.annotation.EnableTransactionManagement;@SpringBootApplication @EnableTransactionManagement @Slf4j @EnableCaching public class SkyApplication { public static void main (String[] args) { SpringApplication.run(SkyApplication.class, args); log.info("server started" ); } }
3). 在用户端接口 SetmealController 的 list 方法上加入 @Cacheable 注解
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 @GetMapping("/list") @ApiOperation("根据分类id查询套餐") @Cacheable(cacheNames = "setmealCache",key = "#categoryId") public Result<List<Setmeal>> list (Long categoryId) { Setmeal setmeal = new Setmeal (); setmeal.setCategoryId(categoryId); setmeal.setStatus(StatusConstant.ENABLE); List<Setmeal> list = setmealService.list(setmeal); return Result.success(list); }
4). 在管理端接口 SetmealController 的 save、delete、update、startOrStop 等方法上加入 CacheEvict 注解
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 @PostMapping @ApiOperation("新增套餐") @CacheEvict(cacheNames = "setmealCache",key = "#setmealDTO.categoryId") public Result save (@RequestBody SetmealDTO setmealDTO) { setmealService.saveWithDish(setmealDTO); return Result.success(); } @DeleteMapping @ApiOperation("批量删除套餐") @CacheEvict(cacheNames = "setmealCache",allEntries = true) public Result delete (@RequestParam List<Long> ids) { setmealService.deleteBatch(ids); return Result.success(); } @PutMapping @ApiOperation("修改套餐") @CacheEvict(cacheNames = "setmealCache",allEntries = true) public Result update (@RequestBody SetmealDTO setmealDTO) { setmealService.update(setmealDTO); return Result.success(); } @PostMapping("/status/{status}") @ApiOperation("套餐起售停售") @CacheEvict(cacheNames = "setmealCache",allEntries = true) public Result startOrStop (@PathVariable Integer status, Long id) { setmealService.startOrStop(status, id); return Result.success(); }
# 功能测试通过前后端联调方式来进行测试,同时观察 redis 中缓存的套餐数据。和缓存菜品 功能测试基本一致,不再赘述。
# 添加购物车# 需求分析和设计# 产品原型用户可以将菜品或者套餐添加到购物车。对于菜品来说,如果设置了口味信息,则需要选择规格后才能加入购物车;对于套餐来说,可以直接点击 将当前套餐加入购物车。在购物车中可以修改菜品和套餐的数量,也可以清空购物车。
效果图:
# 接口设计通过上述原型图,设计出对应的添加购物车接口。
说明:
添加购物车时,有可能添加菜品,也有可能添加套餐。故传入参数要么是菜品 id,要么是套餐 id。
# 表设计用户的购物车数据,也是需要保存在数据库中的,购物车对应的数据表为 shopping_cart 表,具体表结构如下:
字段名 数据类型 说明 备注 id bigint 主键 自增 name varchar(32) 商品名称 冗余字段 image varchar(255) 商品图片路径 冗余字段 user_id bigint 用户 id 逻辑外键 dish_id bigint 菜品 id 逻辑外键 setmeal_id bigint 套餐 id 逻辑外键 dish_flavor varchar(50) 菜品口味 number int 商品数量 amount decimal(10,2) 商品单价 冗余字段 create_time datetime 创建时间
说明:
购物车数据是关联用户的,在表结构中,我们需要记录,每一个用户的购物车数据是哪些 菜品列表展示出来的既有套餐,又有菜品,如果用户选择的是套餐,就保存套餐 ID (setmeal_id),如果用户选择的是菜品,就保存菜品 ID (dish_id) 对同一个菜品 / 套餐,如果选择多份不需要添加多条记录,增加数量 number 即可 # 代码开发# DTO 设计根据添加购物车接口的参数设计 DTO:
在 sky-pojo 模块,ShoppingCartDTO.java 已定义
1 2 3 4 5 6 7 8 9 10 11 12 13 package com.sky.dto;import lombok.Data;import java.io.Serializable;@Data public class ShoppingCartDTO implements Serializable { private Long dishId; private Long setmealId; private String dishFlavor; }
# Controller 层根据添加购物车接口创建 ShoppingCartController:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 package com.sky.controller.user;import com.sky.dto.ShoppingCartDTO;import com.sky.result.Result;import io.swagger.annotations.Api;import io.swagger.annotations.ApiOperation;import lombok.extern.slf4j.Slf4j;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.PostMapping;import org.springframework.web.bind.annotation.RequestBody;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;@RestController @RequestMapping("/user/shoppingCart") @Slf4j @Api(tags = "C端-购物车接口") public class ShoppingCartController { @Autowired private ShoppingCartService shoppingCartService; @PostMapping("/add") @ApiOperation("添加购物车") public Result<String> add (@RequestBody ShoppingCartDTO shoppingCartDTO) { log.info("添加购物车:{}" , shoppingCartDTO); shoppingCartService.addShoppingCart(shoppingCartDTO); return Result.success(); } }
# Service 层接口创建 ShoppingCartService 接口:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 package com.sky.service;import com.sky.dto.ShoppingCartDTO;import com.sky.entity.ShoppingCart;import java.util.List;public interface ShoppingCartService { void addShoppingCart (ShoppingCartDTO shoppingCartDTO) ; }
# Service 层实现类创建 ShoppingCartServiceImpl 实现类,并实现 add 方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 package com.sky.service.impl;import com.sky.context.BaseContext;import com.sky.dto.ShoppingCartDTO;import com.sky.entity.Dish;import com.sky.entity.Setmeal;import com.sky.entity.ShoppingCart;import com.sky.mapper.DishMapper;import com.sky.mapper.SetmealMapper;import com.sky.service.ShoppingCartService;import org.springframework.beans.BeanUtils;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import java.time.LocalDateTime;import java.util.List;@Service public class ShoppingCartServiceImpl implements ShoppingCartService { @Autowired private ShoppingCartMapper shoppingCartMapper; @Autowired private DishMapper dishMapper; @Autowired private SetmealMapper setmealMapper; public void addShoppingCart (ShoppingCartDTO shoppingCartDTO) { ShoppingCart shoppingCart = new ShoppingCart (); BeanUtils.copyProperties(shoppingCartDTO, shoppingCart); shoppingCart.setUserId(BaseContext.getCurrentId()); List<ShoppingCart> shoppingCartList = shoppingCartMapper.list(shoppingCart); if (shoppingCartList != null && shoppingCartList.size() == 1 ) { shoppingCart = shoppingCartList.get(0 ); shoppingCart.setNumber(shoppingCart.getNumber() + 1 ); shoppingCartMapper.updateNumberById(shoppingCart); } else { Long dishId = shoppingCartDTO.getDishId(); if (dishId != null ) { Dish dish = dishMapper.getById(dishId); shoppingCart.setName(dish.getName()); shoppingCart.setImage(dish.getImage()); shoppingCart.setAmount(dish.getPrice()); } else { Setmeal setmeal = setmealMapper.getById(shoppingCartDTO.getSetmealId()); shoppingCart.setName(setmeal.getName()); shoppingCart.setImage(setmeal.getImage()); shoppingCart.setAmount(setmeal.getPrice()); } shoppingCart.setNumber(1 ); shoppingCart.setCreateTime(LocalDateTime.now()); shoppingCartMapper.insert(shoppingCart); } } }
# Mapper 层创建 ShoppingCartMapper 接口:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 package com.sky.mapper;import com.sky.entity.ShoppingCart;import org.apache.ibatis.annotations.Delete;import org.apache.ibatis.annotations.Insert;import org.apache.ibatis.annotations.Mapper;import org.apache.ibatis.annotations.Update;import java.util.List;@Mapper public interface ShoppingCartMapper { List<ShoppingCart> list (ShoppingCart shoppingCart) ; @Update("update shopping_cart set number = #{number} where id = #{id}") void updateNumberById (ShoppingCart shoppingCart) ; @Insert("insert into shopping_cart (name, user_id, dish_id, setmeal_id, dish_flavor, number, amount, image, create_time) " + " values (#{name},#{userId},#{dishId},#{setmealId},#{dishFlavor},#{number},#{amount},#{image},#{createTime})") void insert (ShoppingCart shoppingCart) ; }
创建 ShoppingCartMapper.xml:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mapper namespace ="com.sky.mapper.ShoppingCartMapper" > <select id ="list" parameterType ="ShoppingCart" resultType ="ShoppingCart" > select * from shopping_cart <where > <if test ="userId != null" > and user_id = #{userId} </if > <if test ="dishId != null" > and dish_id = #{dishId} </if > <if test ="setmealId != null" > and setmeal_id = #{setmealId} </if > <if test ="dishFlavor != null" > and dish_flavor = #{dishFlavor} </if > </where > order by create_time desc </select > </mapper >
# 功能测试进入小程序,添加菜品
加入购物车,查询数据库
因为现在没有实现查看购物车功能,所以只能在表中进行查看。
在前后联调时,后台可通断点方式启动,查看运行的每一步。
# 查看购物车# 需求分析和设计# 产品原型当用户添加完菜品和套餐后,可进入到购物车中,查看购物中的菜品和套餐。
# 接口设计
# 代码开发# Controller 层在 ShoppingCartController 中创建查看购物车的方法:
1 2 3 4 5 6 7 8 9 @GetMapping("/list") @ApiOperation("查看购物车") public Result<List<ShoppingCart>> list () { return Result.success(shoppingCartService.showShoppingCart()); }
# Service 层接口在 ShoppingCartService 接口中声明查看购物车的方法:
1 2 3 4 5 List<ShoppingCart> showShoppingCart () ;
# Service 层实现类在 ShoppingCartServiceImpl 中实现查看购物车的方法:
1 2 3 4 5 6 7 8 9 10 public List<ShoppingCart> showShoppingCart () { return shoppingCartMapper.list(ShoppingCart. builder(). userId(BaseContext.getCurrentId()). build()); }
# 功能测试当进入小程序时,就会发起查看购物车的请求
点击购物车图标
测试成功。
# 清空购物车# 需求分析和设计# 产品原型当点击清空按钮时,会把购物车中的数据全部清空。
# 接口设计
# 代码开发# Controller 层在 ShoppingCartController 中创建清空购物车的方法:
1 2 3 4 5 6 7 8 9 10 @DeleteMapping("/clean") @ApiOperation("清空购物车商品") public Result<String> clean () { shoppingCartService.cleanShoppingCart(); return Result.success(); }
# Service 层接口在 ShoppingCartService 接口中声明清空购物车的方法:
1 2 3 4 void cleanShoppingCart () ;
# Service 层实现类在 ShoppingCartServiceImpl 中实现清空购物车的方法:
1 2 3 4 5 6 public void cleanShoppingCart () { shoppingCartMapper.deleteByUserId(BaseContext.getCurrentId()); }
# Mapper 层在 ShoppingCartMapper 接口中创建删除购物车数据的方法:
1 2 3 4 5 6 7 @Delete("delete from shopping_cart where user_id = #{userId}") void deleteByUserId (Long userId) ;
# 功能测试进入到购物车页面
点击清空
查看数据库中的数据
说明当前用户的购物车数据已全部删除。
# 购物车其他功能代码# 删除购物车中一个商品# 产品原型
# 接口设计
# 数据模型shopping_cart 表:
# 代码开发
1 2 3 4 5 6 7 8 9 10 11 12 @PostMapping("/sub") @ApiOperation("删除购物车中一个商品") public Result sub (@RequestBody ShoppingCartDTO shoppingCartDTO) { log.info("删除购物车中一个商品,商品:{}" , shoppingCartDTO); shoppingCartService.subShoppingCart(shoppingCartDTO); return Result.success(); }
1 2 3 4 5 void subShoppingCart (ShoppingCartDTO shoppingCartDTO) ;
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 public void subShoppingCart (ShoppingCartDTO shoppingCartDTO) { ShoppingCart shoppingCart = new ShoppingCart (); BeanUtils.copyProperties(shoppingCartDTO,shoppingCart); shoppingCart.setUserId(BaseContext.getCurrentId()); List<ShoppingCart> list = shoppingCartMapper.list(shoppingCart); if (list != null && list.size() > 0 ){ shoppingCart = list.get(0 ); Integer number = shoppingCart.getNumber(); if (number == 1 ){ shoppingCartMapper.deleteById(shoppingCart.getId()); }else { shoppingCart.setNumber(shoppingCart.getNumber() - 1 ); shoppingCartMapper.updateNumberById(shoppingCart); } } }
1 2 3 4 5 6 @Delete("delete from shopping_cart where id = #{id}") void deleteById (Long id) ;