在项目开发过程中,往往会遇到多级菜单、分类等多层级结构数据的查询。本文就来为大家讲讲MyBatis实现递归查询的方法,感兴趣的可以动手尝试一下
业务场景
在项目开发过程中,往往会遇到多级菜单、分类等多层级结构数据的查询。
for example:
请看上图,这是一个电商项目中常见的多级类目功能,如图所示,共分为一、二、三,共三级类目,每一个一级类目有各自的二级目录,每个二级目录有自己的三级目录
再看这个例子,下图是一个多级菜单的功能,和上面的例子类似
在这种场景下,通常我们要用递归进行处理。下面博主以电商项目的多级类目功能,通过MyBatis进行递归查询功能说明
开发环境
名称 | 版本 |
---|---|
IntelliJ IDEA | 2021.3.2 |
Spring Boot | 2.6.4 |
mybatis-spring-boot-starter | 2.2.2 |
数据库
表结构
CREATE TABLE `shopping_commodity_category`
(
`id` int NOT NULL AUTO_INCREMENT,
`name` varchar(255) NOT NULL COMMENT 'category name',
`picture` varchar(255) DEFAULT NULL COMMENT 'category picture id',
`superior_category` int NOT NULL DEFAULT 0 COMMENT 'superior category id',
`sort_number` int NOT NULL COMMENT 'sort number',
`create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'creation time of this record',
`update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT 'update time of this record',
PRIMARY KEY (`id`),
UNIQUE KEY (`superior_category`, `name`),
UNIQUE KEY (`superior_category`, `sort_number`)
);
字段解释:
字段名 | 含义 |
---|---|
id | 数据表中每条数据的唯一标识符 |
name | 类目名称 |
picture | 类目的预览图id(由于业务需要,这个地方保存的另一张表的文件的id,可根据需要直接存储图片地址),由于上级父分类没有预览图,所以可以为空 |
superior_category | 上级类目的id,不能为空,如果是顶级类目,那么他的上级类目id就为0 |
sort_number | 排序号,用于在同一级的目录下进行排序 |
create_time | 数据的创建时间,无需关注,值是插入数据时自动通过当前时间戳填充 |
update_time | 数据的更新时间,无需关注,值是在当这条记录被更新时自动以当前时间戳进行更新 |
约束解释:
类型 | 用途 |
主键约束(id) | (显而易见,无需赘述) |
联合唯一约束(UNIQUE KEY (superior_category, name)) | 在同一级类目下面,限制不能有重复的类目名称 |
联合唯一约束(UNIQUE KEY (superior_category, sort_number)) | 在同一级类目下,限制排序号不能重复 |
由于博主这个地方业务需要,用到了另一张表的数据,为了说明问题,博主将另一张表的结构也贴出来参考
CREATE TABLE `storage_multimedia_file`
(
`id` int NOT NULL AUTO_INCREMENT,
`absolute_path` text NOT NULL COMMENT 'file absolute path',
`create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'creation time of this record',
`update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT 'update time of this record',
PRIMARY KEY (`id`)
);
这个表很简单,就一个主要的字段,存储的这个文件的绝对路径,方便后续通过IO流读取
插入测试数据
shopping_commodity_category表
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (1, '手机通讯', null, 0, 1, '2022-08-11 17:27:15', '2022-08-11 17:27:15');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (2, '手机配件', null, 1, 1, '2022-08-11 17:29:59', '2022-08-11 17:29:59');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (3, '手机耳机', '2', 2, 1, '2022-08-11 17:29:59', '2022-08-11 17:29:59');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (4, '蓝牙耳机', '3', 2, 2, '2022-08-11 17:31:26', '2022-08-11 17:31:26');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (5, '手机壳/保护壳', '4', 2, 3, '2022-08-11 17:32:51', '2022-08-11 17:32:51');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (6, '手机贴膜', '5', 2, 4, '2022-08-11 17:33:44', '2022-08-11 17:33:44');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (7, '运营商', null, 1, 2, '2022-08-11 17:34:44', '2022-08-11 17:34:44');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (8, '办号卡', '6', 7, 1, '2022-08-11 17:36:25', '2022-08-11 17:36:25');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (9, '家用电器', null, 0, 2, '2022-08-11 17:36:54', '2022-08-11 17:36:54');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (10, '生活电器', null, 9, 1, '2022-08-11 17:37:46', '2022-08-11 17:37:46');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (11, '吸尘器', '7', 10, 1, '2022-08-11 17:38:55', '2022-08-11 17:38:55');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (12, '厨房小电', null, 9, 2, '2022-08-11 17:40:04', '2022-08-11 17:40:04');
INSERT INTO shopping_commodity_category (id, name, picture, superior_category, sort_number, create_time, update_time) VALUES (13, '电饭煲', '8', 12, 1, '2022-08-11 17:41:17', '2022-08-11 17:41:17');
storage_multimedia_file表
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (1, 'D:\\files\\trembling-bird\\commodity-previews\\1.webp', '2022-08-11 16:18:49', '2022-08-11 16:18:49');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (2, 'D:\\files\\trembling-bird\\category-preview\\1.jpg', '2022-08-11 17:30:12', '2022-08-11 17:30:12');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (3, 'D:\\files\\trembling-bird\\category-preview\\2.jpg', '2022-08-11 17:31:07', '2022-08-11 17:31:07');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (4, 'D:\\files\\trembling-bird\\category-preview\\3.png', '2022-08-11 17:32:07', '2022-08-11 17:32:07');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (5, 'D:\\files\\trembling-bird\\category-preview\\4.jpg', '2022-08-11 17:33:36', '2022-08-11 17:33:36');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (6, 'D:\\files\\trembling-bird\\category-preview\\5.png', '2022-08-11 17:35:48', '2022-08-11 17:35:48');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (7, 'D:\\files\\trembling-bird\\category-preview\\6.jpg', '2022-08-11 17:38:22', '2022-08-11 17:38:22');
INSERT INTO trembling_bird.storage_multimedia_file (id, absolute_path, create_time, update_time) VALUES (8, 'D:\\files\\trembling-bird\\category-preview\\7.jpg', '2022-08-11 17:40:38', '2022-08-11 17:40:38');
关键代码
实体类代码
实体类的公共父类(因为数据库中每个表都有共同的字段,如id,create_time,update_time,所以将这些共有的字段抽取出来放到公共的父类,让其他实体类继承此父类,就有了这些公共字段,降低代码冗余)
package com.fenzhimedia.commons.pojo;
import lombok.Data;
import java.io.Serializable;
import java.time.LocalDateTime;
/**
* @author Yi Dai 484201132@qq.com
* @since 2022/3/21 13:55
*/
@Data
public abstract class BasePojo implements Serializable {
/**
* the unique identification of this record in the database table
*/
protected int id;
/**
* creation time of this record
*/
protected LocalDateTime createTime;
/**
* update time of this record
*/
protected LocalDateTime updateTime;
}
描述类目的实体类
package com.fenzhimedia.commons.shopping.pojo;
import com.fenzhimedia.commons.pojo.BasePojo;
import com.fenzhimedia.commons.storage.pojo.MultimediaFile;
import lombok.Data;
import lombok.EqualsAndHashCode;
import java.util.List;
/**
* @author Yi Dai 484201132@qq.com
* @since 2022/4/23 10:43
*/
@Data
@EqualsAndHashCode(callSuper = true)
public class CommodityCategory extends BasePojo {
/**
* category name
*/
private String name;
/**
* entity class encapsulating picture information
*/
private MultimediaFile picture;
/**
* used to specify the order of categories,
* which is only valid under the same level category
*/
private Integer sortNumber;
/**
* sub commodity category
*/
private List<CommodityCategory> subCommodityCategories;
}
描述文件的实体类
package com.fenzhimedia.commons.storage.pojo;
import com.fenzhimedia.commons.pojo.BasePojo;
import lombok.Data;
import lombok.EqualsAndHashCode;
/**
* @author Yi Dai 484201132@qq.com
* @since 2022/8/11 16:03
*/
@Data
@EqualsAndHashCode(callSuper = true)
public class MultimediaFile extends BasePojo {
private String absolutePath;
}
MyBatis的mapper接口代码
package com.fenzhimedia.shopping.mapper;
import com.fenzhimedia.commons.shopping.pojo.CommodityCategory;
import java.util.List;
/**
* @author Yi Dai 484201132@qq.com
* @since 2022/8/11 16:46
*/
public interface CommodityCategoryMapper {
List<CommodityCategory> queryCommodityCategories(int superiorCategoryId);
}
mapper映射文件代码
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.fenzhimedia.shopping.mapper.CommodityCategoryMapper">
<select id="queryCommodityCategories" resultMap="commodityCategoryMap">
select `shopping_commodity_category`.`id` as `shopping_commodity_category_id`,
`shopping_commodity_category`.`name` as `shopping_commodity_category_name`,
`shopping_commodity_category`.`picture` as `shopping_commodity_category_picture`,
`shopping_commodity_category`.`superior_category` as `shopping_commodity_category_superior_category`,
`storage_multimedia_file`.`id` as `storage_multimedia_file_id`,
`storage_multimedia_file`.`absolute_path` as `storage_multimedia_file_absolute_path`
from `shopping_commodity_category`
left join `storage_multimedia_file`
on `shopping_commodity_category`.`picture` = `storage_multimedia_file`.`id`
where `superior_category` = #{categoryId}
order by `shopping_commodity_category`.`sort_number`
</select>
<resultMap id="commodityCategoryMap" type="commodityCategory">
<id property="id" column="shopping_commodity_category_id"/>
<result property="name" column="shopping_commodity_category_name"/>
<association property="picture" javaType="multimediaFile">
<id property="id" column="storage_multimedia_file_id"/>
<result property="absolutePath" column="storage_multimedia_file_absolute_path"/>
</association>
<collection property="subCommodityCategories"
ofType="commodityCategory"
column="shopping_commodity_category_id"
select="queryCommodityCategories"/>
</resultMap>
</mapper>
代码解释:
可以看到,queryCommodityCategories是一个类目表和文件表的左连接查询,然后分别起了别名,接收一个int型的参数,为上级类目的id,当然也就是0,然后声明一个resultMap ,将字段映射起来,都是基本操作,唯一值得注意的就是
subCommodityCategories作为一个集合,它有通过select直接调用了queryCommodityCategories这个查询,而参数正是由column属性传递过去的,参数的值就是当前类目的id(shopping_commodity_category_id)(有点绕),其实这样就递归查询起来了
那么有的小伙伴可能会疑问,既然是已知是0,为何不直接写道xml中,而是大费周章的,在接口上声明一个参数,然后传递进来?其实这是因为后面的递归查询的时候需要传入上级分类的id,一旦写死了,就只能查询上级分类id为0的,很显然,达不到想要的效果
既然如此,那么就需要业务代码中来传递上级类目id这个参数,很显然我们直接在代码中写死不是那么的优雅。博主这里为了更灵活,我想要实现一个如果前台传递了上级分类id,那么就帮他查询指定上级类目的子级类目,如果没有传递,那么就查询所有的类目及其子类目。所以博主是这么处理的:
@GetMapping("/queryCommodityCategories")
public ResponseBody queryCommodityCategories(@RequestParam(required = false, defaultValue = "0") int superiorCategoryId) {
return commodityCategoryService.queryCommodityCategories(superiorCategoryId);
}
查询测试
返回结果
statusCode、message是通用返回实体类的结构,无需关注,查询的数据在data中
{
"statusCode": 200,
"message": null,
"data": [
{
"id": 1,
"createTime": null,
"updateTime": null,
"name": "手机通讯",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 2,
"createTime": null,
"updateTime": null,
"name": "手机配件",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 3,
"createTime": null,
"updateTime": null,
"name": "手机耳机",
"picture": {
"id": 2,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\1.jpg"
},
"sortNumber": null,
"subCommodityCategories": [ ]
},
{
"id": 4,
"createTime": null,
"updateTime": null,
"name": "蓝牙耳机",
"picture": {
"id": 3,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\2.jpg"
},
"sortNumber": null,
"subCommodityCategories": [ ]
},
{
"id": 5,
"createTime": null,
"updateTime": null,
"name": "手机壳/保护壳",
"picture": {
"id": 4,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\3.png"
},
"sortNumber": null,
"subCommodityCategories": [ ]
},
{
"id": 6,
"createTime": null,
"updateTime": null,
"name": "手机贴膜",
"picture": {
"id": 5,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\4.jpg"
},
"sortNumber": null,
"subCommodityCategories": [ ]
}
]
},
{
"id": 7,
"createTime": null,
"updateTime": null,
"name": "运营商",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 8,
"createTime": null,
"updateTime": null,
"name": "办号卡",
"picture": {
"id": 6,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\5.png"
},
"sortNumber": null,
"subCommodityCategories": [ ]
}
]
}
]
},
{
"id": 9,
"createTime": null,
"updateTime": null,
"name": "家用电器",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 10,
"createTime": null,
"updateTime": null,
"name": "生活电器",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 11,
"createTime": null,
"updateTime": null,
"name": "吸尘器",
"picture": {
"id": 7,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\6.jpg"
},
"sortNumber": null,
"subCommodityCategories": [ ]
}
]
},
{
"id": 12,
"createTime": null,
"updateTime": null,
"name": "厨房小电",
"picture": null,
"sortNumber": null,
"subCommodityCategories": [
{
"id": 13,
"createTime": null,
"updateTime": null,
"name": "电饭煲",
"picture": {
"id": 8,
"createTime": null,
"updateTime": null,
"absolutePath": "D:\\files\\trembling-bird\\category-preview\\7.jpg"
},
"sortNumber": null,
"subCommodityCategories": [ ]
}
]
}
]
}
]
}
以上就是MyBatis实现递归查询的方法详解的详细内容,更多关于MyBatis递归查询的资料请关注编程学习网其它相关文章!
本文标题为:MyBatis实现递归查询的方法详解
基础教程推荐
- java实现多人聊天系统 2023-05-19
- Java实现查找文件和替换文件内容 2023-04-06
- ConditionalOnProperty配置swagger不生效问题及解决 2023-01-02
- JDK数组阻塞队列源码深入分析总结 2023-04-18
- java基础知识之FileInputStream流的使用 2023-08-11
- Java实现线程插队的示例代码 2022-09-03
- Java并发编程进阶之线程控制篇 2023-03-07
- Java数据结构之对象比较详解 2023-03-07
- Java文件管理操作的知识点整理 2023-05-19
- springboot自定义starter方法及注解实例 2023-03-31