加载酷鲨商城前台数据库
在给大家提供的csmall-jsd2203项目的doc文件夹下的sql文件夹中
有多个sql语句文件
分别去运行它们,我们可以获得酷鲨商城前台的数据库信息了
我们每个微服务项目原则上只操作少于一个数据库
开发分类功能
分类功能实现逻辑
我们数据库mall_pms的category表使用自关联实现了三级分类
当前酷鲨商城项目使用固定的三级分类
1.从数据库中查询出所有分类信息,一次性全查
2.构建分类信息的父子结构,实现查询返回父子结构的分类信息
3.将查询到的结果保存在Redis中,以备后续用户直接获取
代码中要判断Redis中是否包含全部分类数据,不包含的话做上面操作
包含分类数据的话直接获得之后返回
业务分析
查询全部分类的业务重点在构建三级分类树结构
我们需要将从数据库中查询出的分类对象构成下面的结构
[
{id:1,name:"手机/运营商/数码",parentId:0,depth:1,children:[
{id:2,name:"手机通讯",parentId:1,depth:2,children:[
{id:3,name:"智能手机",parentId:2,depth:3,children:null},
{id:4,name:"非智能手机",parentId:2,depth:3,children:null}
]},
]},
{id:5,name:"电脑/办公",parentId:0,depth:1,children:[....]}
]
上面是我们需要获得的对象的结构
可以理解为下图
在数据库mall_pms中
有pms_category表,这个表就是保存全部分类信息的表格
id:主键
name:显示在页面上的分类名称
parentId:父分类的id 如果是一级分类父分类id为0
depth:分类深度,当前项目就是3级分类,1\2\3 分别代表它的等级
keyword:搜索关键字
sort:排序依据 正常查询时,根据此列进行排序,数字越小越出现在前面(升序)
icon:图标地址
enable:是否可用
isparent:是否为父分类 0 假 1真
isdisplay:是否显示在导航栏 0不显示 1显示
实施开发
front:前台
在csmall-front-webapi项目中开发
无需编写持久层代码,因为mall_pms数据库的所有操作均在product模块编写完成了
我们front模块只需dubbo调用即可
创建service.impl包
包中编写业务逻辑层实现类FrontCategoryServiceImpl 实现IFrontCategoryService
实施开发
业务逻辑层最终代码如下
@Service
@Slf4j
public class FrontCategoryServiceImpl implements IFrontCategoryService {
// front模块要dubbo调用product模块的方法,实现查询所有分类信息列表
@DubboReference
private IForFrontCategoryService dubboCategoryService;
// 方法要将查询到的分类信息保存到Redis,所以需要操作redis的对象
@Autowired
private RedisTemplate redisTemplate;
// 开发时,使用Redis要定义一个常量,作为key的名称,防止编码是拼写错误
public static final String CATEGORY_TREE_KEY="category_tree";
// 返回三级分类树对象
@Override
public FrontCategoryTreeVO categoryTree() {
// 方法开始,先检查redis中是否已经包含这个key
if(redisTemplate.hasKey(CATEGORY_TREE_KEY)){
// 当Redis中包含这个key时,从redis中获取对应的值直接返回即可
FrontCategoryTreeVO<FrontCategoryEntity> treeVO=
(FrontCategoryTreeVO<FrontCategoryEntity>)
redisTemplate.boundValueOps(CATEGORY_TREE_KEY).get();
// 别忘了返回
return treeVO;
}
// Redis中没有三级分类树信息,表示本次请求可能是第一次访问
// 这样就需要从数据库查询所有分类信息,构建为三级分类树对象,再保存到Redis
// 利用dubbo调用product模块查询所有分类信息的功能
List<CategoryStandardVO> categoryList=
dubboCategoryService.getCategoryList();
// 上面查询到的categoryList就包含了所有的分类信息
// 下面要将分类信息按照级别构建成分类树,需要一个能够包含保存子分类集合属性的类(children属性)
// FrontCategoryEntity这个类就是包含分类基本属性和子分类集合属性的实体类
// 因为转换过程比较复杂,所以单独编写一个方法实现转换
FrontCategoryTreeVO<FrontCategoryEntity> treeVO=initTree(categoryList);
// 上面方法完成了三级分类树的构建,返回了treeVO
// 下面要将treeVO保存到Redis,方便后面请求获取
redisTemplate.boundValueOps(CATEGORY_TREE_KEY).set(
treeVO,
1,
TimeUnit.MINUTES);
// 上面设置了redis中数据有效期,学习和测试过程中不要设置太长,这里1分钟即可
// 实际开发中要设置时间比较长,例如24小时甚至更长
// 这里要返回treeVO!!!!!
return treeVO;
}
private FrontCategoryTreeVO<FrontCategoryEntity> initTree(
List<CategoryStandardVO> categoryList) {
// 第一步:
// 确定所有分类对象对应的父分类id
// 创建一个Map
// 以父分类id为key,将当前正在遍历的分类对象,保存在对应的value中
Map<Long,List<FrontCategoryEntity>> map=new HashMap<>();
log.info("准备开始构建三级分类树,分类树元素总数:{}",categoryList.size());
// 遍历数据库查询出的所有分类对象集合
for(CategoryStandardVO categoryStandardVO : categoryList){
// 当前正在遍历的categoryStandardVO对象,没有children属性
// 需要转换为有children属性的FrontCategoryEntity
FrontCategoryEntity frontCategoryEntity=new FrontCategoryEntity();
// 将categoryStandardVO同名属性赋值到frontCategoryEntity对象
BeanUtils.copyProperties(categoryStandardVO,frontCategoryEntity);
// 因为后面会多次使用到父分类id,所以这个提取出来
Long parentId=frontCategoryEntity.getParentId();
// 判断map中是否已经存在这个父分类id作为key的元素
if(!map.containsKey(parentId)){
// 如果map中没有这个key,表示当前分类对象的父分类id第一次出现
// 要在map中新增个元素,key就是这个父分类id,元素的值是个list要实例化
List<FrontCategoryEntity> value=new ArrayList<>();
// 将当前分类对象保存到这个list中
value.add(frontCategoryEntity);
// 最后将key和value组合保存到map
map.put(parentId,value);
}else{
// 如果map中已经存在当前分类对象的父分类id为key的元素
// 我们就直接获取这个key的value,在value中添加当前分类对象
map.get(parentId).add(frontCategoryEntity);
}
}
// 第二步:
// 构建三级分类树,将子分类对象集合添加到对应的父分类对象的children属性中
// 先从所有的一级分类对象开始,已经是父分类id为0的对象
List<FrontCategoryEntity> firstLevels=map.get(0L);
// 判断一级分类集合是否为null(或元素个数为0),抛出异常终止程序
if(firstLevels == null || firstLevels.isEmpty()){
throw new CoolSharkServiceException(
ResponseCode.INTERNAL_SERVER_ERROR,"没有一级分类对象!");
}
// 遍历一级分类集合
for(FrontCategoryEntity oneLevel : firstLevels){
// 一级分类对象的id就是二级分类对象的父id
Long secondLevelParentId=oneLevel.getId(); // getId()!!!!!!!!!!!!!!!
// 根据二级分类的父Id获取二级分类对象集合
List<FrontCategoryEntity> secondLevels=map.get(secondLevelParentId);
// 判断二级分类集合是否为null
if(secondLevels == null || secondLevels.isEmpty()){
// 二级分类集合缺失,不用抛异常,只需在日志输出警告即可
log.warn("当前二级分类没有内容:{}",secondLevelParentId);
// 当前二级分类没有内容,无需运行循环中后面的语句,直接进行下次循环
continue;
}
// 二级分类集合确认有元素,可以开始编写二级分类集合
for(FrontCategoryEntity twoLevel : secondLevels){
// 获取当前二级分类的id,作为三级分类的父id
Long thirdLevelParentId=twoLevel.getId(); // getId()!!!!!!!!!!!!!!!
// 根据这个thirdLevelParentId获取所有三级分类对象集合
List<FrontCategoryEntity> thirdLevels=
map.get(thirdLevelParentId);
// 还是判断三级分类对象是否为null
if(thirdLevels == null || thirdLevels.isEmpty()){
log.warn("当前二级分类没有三级分类内容:{}",thirdLevelParentId);
continue;
}
// 三级分类集合不是至少有一个元素,将它赋值给二级分类的children属性
twoLevel.setChildrens(thirdLevels);
}
// 在内层循环结束后,在外层循环结束前
// 将二级分类集合赋值给一级分类对象的children属性
oneLevel.setChildrens(secondLevels);
}
// 循环结束后,我们所有的分类对象都已经保存在了自己对应的父分类对象的children属性中了
// 但是我们最终返回的是FrontCategoryTreeVO类型
// 将一级分类对象集合,保存到FrontCategoryTreeVO类型中返回
FrontCategoryTreeVO<FrontCategoryEntity> treeVO=
new FrontCategoryTreeVO<>();
treeVO.setCategories(firstLevels);
// 最后千万别忘了返回treeVO
return treeVO;
}
}
创建控制层
controller包
CategoryController类
代码如下
@RestController
@RequestMapping("/front/category")
@Api(tags = "前台分类查询")
public class CategoryController {
@Autowired
private IFrontCategoryService categoryService;
@GetMapping("/all")
@ApiOperation("查询获取三级