MyBatisPlus自定义sql
在说怎么实现之前我们要先明白一个概念,就是mybatis-plus是在mybatis的基础上进行增强,并不做改变,所以mybatis的操作在mybatis-plus中也是一样可以使用的,咱们直接上代码.
1、单纯地使用注解自定义SQL
data:image/s3,"s3://crabby-images/08127/08127c760f35a250e3a867491746b79946076daf" alt=""
data:image/s3,"s3://crabby-images/2cf86/2cf8626be301efdfe1579cc6d7e2a6cabcdded66" alt=""
data:image/s3,"s3://crabby-images/b54d2/b54d262489dbd223ef6dd7af9780a709b488aa17" alt=""
data:image/s3,"s3://crabby-images/3ee98/3ee989d845bd9b85901c1d081c09388a842e5e1a" alt=""
data:image/s3,"s3://crabby-images/a950e/a950e4a66444b39c42b9b58c191e3f3895050a40" alt=""
data:image/s3,"s3://crabby-images/13b72/13b7217a908b0ddf885c4fc5132bdf11a711bba9" alt=""
data:image/s3,"s3://crabby-images/2e725/2e725f1f79342e20c812845f7ad0ff41a22c0234" alt=""
data:image/s3,"s3://crabby-images/b4d17/b4d17a18e18c8bc83e930fecd5003fc53572dd31" alt=""
数据库数据:
data:image/s3,"s3://crabby-images/9b62d/9b62dd9d7059152b0560129ad96dc664a88fdf3c" alt=""
执行测试方法后结果如下:
data:image/s3,"s3://crabby-images/174fd/174fd1f13057c41cf3db3c4d3b71d37f0633e623" alt=""
注意:这里我们把MybatisPlusConfig这个类注释掉,持久层的类使用@Mapper注解,也是一样可以运行的哈。
data:image/s3,"s3://crabby-images/db841/db841744f5fe0ca7e03bcaf786c462b0099cc573" alt=""
data:image/s3,"s3://crabby-images/83ca9/83ca9d5ef4b25b554dd25fdfe8161cfb63a8c811" alt=""
结果:
data:image/s3,"s3://crabby-images/1b534/1b534a93a2b19a261bb71ec16bfbad8f885daccb" alt=""
2、使用注解加标签做自定义sql
展示1:使用if标签
data:image/s3,"s3://crabby-images/4c8f7/4c8f79f5a9041f5ba051ca3fc596ad0c6401cff3" alt=""
data:image/s3,"s3://crabby-images/34082/340820e56a4ae9f0009d275bff8f9926603a1af4" alt=""
data:image/s3,"s3://crabby-images/7fd0f/7fd0fc1c73d641dd933c7d8ed11ff3f64d3caf48" alt=""
data:image/s3,"s3://crabby-images/374e8/374e8e2f1dbdaa2cea57d3478f97d5ae1872235a" alt=""
结果:
data:image/s3,"s3://crabby-images/24c37/24c377e2de284fa091abb2905e623fc1c5b93058" alt=""
展示2:使用foreach标签
data:image/s3,"s3://crabby-images/ef21d/ef21d89f4a45f1189f9107010f82b1671ff69a3a" alt=""
data:image/s3,"s3://crabby-images/665d2/665d2cfe69a7325e09410b6def61a3239e2b7813" alt=""
data:image/s3,"s3://crabby-images/ac606/ac60642b5fe94a410fef50d8a98a030cc1281f57" alt=""
data:image/s3,"s3://crabby-images/05640/0564068aa52cbf7144ae34c9dee0e54d63ac1563" alt=""
结果:
data:image/s3,"s3://crabby-images/036f2/036f2be5fda0bdecf2dc9ab0df55f2d6a66db4b3" alt=""
展示3:使用foreach标签
下面我们使用id来查找一下:
data:image/s3,"s3://crabby-images/c79b7/c79b7de2bea1f58e29f38992de1172c480d85d20" alt=""
data:image/s3,"s3://crabby-images/a8dec/a8dec6947f286737ca4baa0a1fe7fc09f6b415f1" alt=""
data:image/s3,"s3://crabby-images/b4206/b4206bb72ef0c3688ddcfa445a68498a9f428bbf" alt=""
data:image/s3,"s3://crabby-images/f115c/f115cb09ac0a833bc8f03264b0cbf67861311b87" alt=""
结果:
data:image/s3,"s3://crabby-images/0bcd4/0bcd467bdacaacd122d03ab8c2469e7ba6ed2d52" alt=""
扩展知识点:{} 和 ${} 的区别?
一、区别概述
1.1、主要区别:
1、#{} 是预编译处理 ,${} 是直接替换;
2、${} 存在SQL注入的问题,而 #{} 不存在;
Ps:这也是面试主要考察的部分~
1.2、细节上:
1、${} 可以实现排序查询,#{} 不能实现排序查询。
2、${} 可以直接进行模糊查询(但不建议,存在 SQL 注入问题) ,#{} 不可以直接进行模糊查询 ,但可以通过 mysql 内置函数 concat() 实现模糊查询(不存在 SQL 注入问题)。
二、具体描述
2.1、预编辑处理 vs 直接替换
预编辑处理:是指 MyBatis 在处理 #{} 时,就是把 #{} 替换成了 ?号,使用 PreparedStatement 的 set 方法来赋值。也就是说 #{} 会把 {} 内的整体看成 value ,最后再给 value 加上单引号,重点强调引号内部是一个整体( #{} 不会发生 SQL 注入的根本原因)。
直接替换:是指 MyBatis 在处理 ${} 时,会把 ${} 替换成变量的值(不会加引号处理)。
2.2、SQL 注入问题
2.2.1、引发 SQL 注入
例如现在有一个登陆程序,需要输入正确的账户和密码才能登录,当使用了 SQL 注入就可以在不知道密码的情况下进行登录,如下
xml 文件如下:
xml
<select id="login" resultType="com.example.demo.entity.Userinfo">
select * from userinfo where username = '${username}' and password = '${password}'
</select>
接口如下:
java
/**
* 登录逻辑
* @param username
* @param password
* @return
*/
Userinfo login(@Param("username") String username,
@Param("password") String password);
测试方法如下:
java
@Test
void login() {
String username = "admin";
String password = "' or 1 = '1";
Userinfo userinfo = userMapper.login(username, password);
System.out.println("登录状态:" + (userinfo == null ? "失败" : "成功"));
}
执行结果如下:
data:image/s3,"s3://crabby-images/6ac63/6ac63026b80cc648a62c2d292438b5e646011557" alt=""
2.2.2、SQL 注入分析
可以在运行结果的日志中看到,我们最后执行的SQL语句如下:
data:image/s3,"s3://crabby-images/7ac1b/7ac1b00746ce1e8ae48888d477f2f7d4f24ade59" alt=""
在 MySQL 中 1 = '1' 结果必然为 true,所以不难分析出,如上 SQL 一定会将这张表中的用户全部返回(即使账号都填写错了,照样返回,因为1 = '1' 必然为 true );
使用 ${} 的注意事项:一定是可以穷举的值,在使用之前一定要对传递的值进行合法性验证(在Controller中通过穷举的方式验证安全性)。
2.3、排序查询
使用 ${} 可以实现排序查询,而 #{} 不可以实现排序查询,因为使用 #{} 查询时,如果传递的值为 String 就会加单引号,导致 sql 错误。
例如你期望的 sql 语句为:select * from userinfo order by id desc;
而 #{sort} 中传入的是一个 String 类型的值为 "desc";
那么最终实际的 sql 语句为:select * from userinfo order by id 'desc';这必然是错误的~
2.4、like 查询
2.4.1、${} 模糊查询
xml
方式一:直接替换
<select id="likeSelect" resultType="com.example.demo.entity.Userinfo">
select * from userinfo where username like '%${key}%'
</select>
方式二:使用concat进行字符串拼接
<select id="likeSelect" resultType="com.example.demo.entity.Userinfo">
select * from userinfo where username like concat('%', '${key}', '%')
</select>
Ps:虽然可以这样,但并不建议使用 ${} 进行模糊查询,因为存在 SQL 注入问题。
2.4.2、#{} 模糊查询
xml
<select id="likeSelect" resultType="com.example.demo.entity.Userinfo">
select * from userinfo where username like concat('%', #{key}, '%')
</select>
2.4.3、#{} 模糊查询问题分析
你期望的 sql 语句:select * from user where username = '%abc%';
然而当你使用 #{} 传入的参数会自带引号,于是就变成了:select * from user where username = '%' abc '%';
注意:
注意一点:接口里面的形参如果没有使用@Param注解,那么你sql里面就不能直接用${}
java
1)使用@Param注解
当以下面的方式进行写SQL语句时:
@Select("select column from table where userid = #{userid} ")
public int selectColumn(int userid);
当你使用了使用@Param注解来声明参数时,如果使用 #{} 或 ${} 的方式都可以。
@Select("select column from table where userid = ${userid} ")
public int selectColumn(@Param("userid") int userid);
当你不使用@Param注解来声明参数时,必须使用使用 #{}方式。如果使用 ${} 的方式,会报错。
@Select("select column from table where userid = ${userid} ")
public int selectColumn(@Param("userid") int userid);
2)不使用@Param注解
不使用@Param注解时,参数只能有一个,并且是Javabean。在SQL语句里可以引用JavaBean的属性,而且只能引用JavaBean的属性。
// 这里id是user的属性
@Select("SELECT * from Table where id = ${id}")
Enchashment selectUserById(User user);
3、注解使用条件构造器来做查询条件
扩展知识点Wrapper中的方法
关于Wrapper中的一些方法,介绍如下:
eq:等于,ne:不等于
gt:大于,ge:大于等于,lt:小于,le:小于等于
between:在值1和值2之间,notBetween:不在值1和值2之间
like:'%值%',notLike:'%值%',likeLeft:'%值',likeRight:'值%'
isNull:字段 IS NULL,isNotNull:字段 IS NOT NULL
in:字段 IN (v0, v1, ...),notIn:字段 NOT IN (value.get(0), value.get(1), ...)
or:拼接 OR,AND 嵌套
注意事项:
主动调用or表示紧接着下一个方法不是用and连接!(不调用or则默认为使用and连接)
exists:拼接 EXISTS ( sql语句 ),notExists:拼接 NOT EXISTS ( sql语句 )
orderByAsc:排序:ORDER BY 字段, ... ASC,orderByDesc:排序:ORDER BY 字段, ... DESC
想要了解详细点可以看:https://blog.csdn.net/llllllkkkkkooooo/article/details/108216957
使用QueryWrapper
演示如下:
data:image/s3,"s3://crabby-images/2b231/2b23104d233328c67729b74b91d51639a80e7260" alt=""
data:image/s3,"s3://crabby-images/f701f/f701f4cb24399cb79b2b11f19894a93f930db130" alt=""
data:image/s3,"s3://crabby-images/b0a66/b0a6651ea5692cb5b2721a341c64b16e139763d3" alt=""
data:image/s3,"s3://crabby-images/52cac/52cacb31ec8b11360f0c49fc9dd375f2efd2b84d" alt=""
效果:
data:image/s3,"s3://crabby-images/dbfcb/dbfcb1f56418db30e09c21455c0d52f968c41997" alt=""
使用LambdaQueryWrapper
上面的这个条件构造器也可以使用LambdaQueryWrapper来做。
比如:
data:image/s3,"s3://crabby-images/7e72e/7e72eb30289f80e904527d1d36b017008fcb06eb" alt=""
效果如下:
data:image/s3,"s3://crabby-images/ec7b9/ec7b9d0aeb380eac8590781388ba14e775855e84" alt=""
e w . s q l S e g m e n t 相当于是拿到条件构造器里面的查询条件们,但是注意:条件构造器中,其实,不止有 {ew.sqlSegment}相当于是拿到条件构造器里面的查询条件们,但是注意:条件构造器中,其实,不止有 ew.sqlSegment相当于是拿到条件构造器里面的查询条件们,但是注意:条件构造器中,其实,不止有{ew.sqlSegment}可以使用。还有 e w . s q l S e l e c t , {ew.sqlSelect}, ew.sqlSelect,{ew.customSqlSegment}, e w . s q l S e t 等等。他们都相当于是 e w 对象中的属性, {ew.sqlSet}等等。他们都相当于是ew对象中的属性, ew.sqlSet等等。他们都相当于是ew对象中的属性,{}只是用来取那些属性的。
关于ew
具体的如下:
-
ew.customSqlSegment相当于是ew中定义的所有查询条件,并且会直接在sql中会在使用的时候先添加 where,即,你sql中可以不用写那个where了
使用例子如下:
java@Select(select * from sys_user ${ew.customSqlSegment}) List<SysUser> listPage(@Param(Constants.WRAPPER) QueryWrapper queryWrapper)
-
ew.sqlSegment属性相当于是ew中定义的所有的查询条件,但是不会加where
使用例子如下:
java@Select(select * from sys_user where ${ew.sqlSegment}) List<SysUser> listPage@Param(Constants.WRAPPER) QueryWrapper queryWrapper)
-
ew.sqlSelect属性相当于是ew中所有你通过queryWrapper.select(......) 所定义查询的字段
java@Select(select ${ew.sqlSelect} from sys_user ) List<SysUser> listPage(@Param(Constants.WRAPPER) QueryWrapper queryWrapper)
xml<select id="selectUser" resultType="com.example.demo.entity.User"> select <if test="ew != null and ew.sqlSelect != null and ew.sqlSelect != ''"> ${ew.sqlSelect} </if> from sys_user where is_deleted != 1 <if test="ew != null"> <if test="ew.nonEmptyOfWhere"> AND </if> ${ew.sqlSegment} </if> </select>
-
ew.sqlSet属性用于update语句
例子:
Mapper接口:
java@Mapper @Repository public interface UserMapper extends BaseMapper<User> { List<User> queryAll(@Param("tableName") String tableName,@Param(Constants.WRAPPER) Wrapper wrapper); boolean updateById(@Param("tableName") String tableName,@Param("id") int id,@Param(Constants.WRAPPER) Wrapper wrapper);//若变量名为ew则无需注解 }
XML:
xml<select id="queryAll" resultType="cn.alan.mybatis.POJO.User"> select ${ew.sqlSelect} from ${tableName} ${ew.customSqlSegment}; </select> <update id="updateById"> update ${tableName} set ${ew.sqlSet} ${ew.customSqlSegment}; </update>
等效select SQL:select * from user where age = 10;
等效update SQL:update user set name = 5 where id = 5;
Controller(或Test):
java@Autowired UserServiceImpl userService; @RequestMapping("/query") public List<User> queryAll(){ QueryWrapper<User> wrapper = new QueryWrapper<>(); wrapper.select("*").eq("age","10"); return userService.queryAll("user",wrapper); //return userService.queryAll("user", Wrappers.query().select("*").eq("age","10")); } @RequestMapping("/update") public boolean upDateById(){ UpdateWrapper<User> wrapper = new UpdateWrapper<>(); wrapper.set("name","5").eq("id","5"); return userService.updateById("user",5,wrapper); //return userService.updateById("user",5,Wrappers.update().set("name","5").eq("id",5)); } }
-
ew.nonEmptyOfNormal这个属性定义在Wrapper类中,用于判断查询条件是否不为空
-
ew.nonEmptyOfWhere这个属性是用于判断你的这个ew中查询条件是否不为空的。
-
ew.emptyOfWhere这个属性定义在Wrapper类中,用于判断where条件是否为空的。
下面具体演示一些上面这些ew属性的使用:
data:image/s3,"s3://crabby-images/51391/513917dbe4b76939c4024c224cea4770c7d8d000" alt=""
data:image/s3,"s3://crabby-images/a808b/a808b8dcdf26d086b410e2e54f5cf386a5722256" alt=""
data:image/s3,"s3://crabby-images/6f9f2/6f9f25400ae54e934ea72c9e0b08e32d390d0345" alt=""
data:image/s3,"s3://crabby-images/62f17/62f17fcd64af50135fef6def3f12fffb6197d864" alt=""
结果:
data:image/s3,"s3://crabby-images/9e503/9e503ee48cb13052bf0c4ebc4917ab31dc9dab93" alt=""
使用条件构造器多表联查
注意:
使用${ew.sqlSegment}的话,如果是多表查询且查询条件是多个表的字段,那么则需在service层拼接查询条件时字段前指定别名,而且不能用lambda的条件构造器来做查询了。因为多表查询的话,如果你两个表里面有需要区分的字段,比如两个表里面都有name,你sql里面的where后面一般写为where user.name='张三' and ......这种的,这个"表名.字段名"这种东西用lambda的条件构造器是写不出来的。但是如果直接用字段名就可以区分的情况,就可以用lambda的条件构造器,没有关系的。
直接用字段名就可以区分的情况,例子:
数据库里面的edu_course如下:
data:image/s3,"s3://crabby-images/f2523/f2523693d1742d3bee0b66f2c310b102eb43acfe" alt=""
data:image/s3,"s3://crabby-images/7f6ff/7f6ffb4d9f23567b60c46ffa1996b22b75daf4f9" alt=""
data:image/s3,"s3://crabby-images/01f40/01f40dcfc78fc8a232b388ccb1824bbdfee44457" alt=""
data:image/s3,"s3://crabby-images/c9c19/c9c197df95333357342f49f695b111a56817755b" alt=""
data:image/s3,"s3://crabby-images/7e603/7e6034d3aafe64131d414ea2536a8ef2dbe1d39c" alt=""
运行,但是结果如下:
data:image/s3,"s3://crabby-images/7bafa/7bafa9b8841bcc77d8877a27824337dd74b9f62d" alt=""
解决方案一:修改数据源的版本
解决方案二:你新建一个实体类来接收返回的数据,不用List<Map<String,Object>>类型来接收了,但是注意你新建的这个实体类要是包含两个表的字段了,因为是多表查询的结果对应的实体类嘛,然后我们在这个新建的实体类里面我们让create_time这个字段被Date类型的实体类属性来接收就行了,就像是单表查询里面我们写的实体类那样。
我们之前单表对应的实体类的接收create_time字段的属性的类型是Date类型的,所以之前没有出现这个问题,如下:
data:image/s3,"s3://crabby-images/1dcd9/1dcd94fbb01ef012b39db462eefdf379744fb639" alt=""
这里我们采用第一种解决方法,修改druid的版本号为1.1.21或者1.1.21以上的版本就行了。
data:image/s3,"s3://crabby-images/a42ff/a42ff0ad5331f725bf218c69caf2ea5a0b8daf7f" alt=""
修改后,运行结果如下:
data:image/s3,"s3://crabby-images/d9d86/d9d867c9662f839870a11a199d0a4f8ecc646aea" alt=""
就相当于是数据库里面是这样的sql:
data:image/s3,"s3://crabby-images/7f97c/7f97ca6cf7d6d26f3f3dd9f2014dee400505c896" alt=""
可以看到sql里面没有需要区分的字段,所以可以用lambda的写法。
但是如果我们把id也作为查询条件呢?
data:image/s3,"s3://crabby-images/d0161/d0161ca0547ec08bc6064aedf181e38cd76ca388" alt=""
因为两个表都有id,所以会查询失败。上面这里就相当于这样的sql执行:
data:image/s3,"s3://crabby-images/475e3/475e35d67abf098f880fa9771d649da0387d81ea" alt=""
看到错误一样。
解决方法就是用"表名.字段名"来表示查询使用哪个表的id作为查询条件。
data:image/s3,"s3://crabby-images/9901f/9901f03702986c8e672b11ba463009790911819a" alt=""
但是LambdaQueryWrapper是根据字段来映射的,是自动的,所以你不好改变查询条件里面的字段名。而且,上面LambdaQueryWrapper这个例子里面,这里你查询结果是有两个id字段的,那么查询结果将会用哪个id放在我们返回对象里面呢?我们看到,上面的例子里面,返回结果里面就封装了edu_teacher表里面的id,而edu_course里面的id是被舍弃了,反正,不管取哪一个,都会有一个id消失了。即,查询结果中名字重复的字段只会保留一个。这样不好。
所以,建议还是用QueryWrapper来做多表的查询条件,不用LambdaQueryWrapper做多表的查询条件。
例子:
data:image/s3,"s3://crabby-images/8230b/8230b07b6b4fff655a7a3111fdce294d3d90b825" alt=""
data:image/s3,"s3://crabby-images/299a4/299a4bac2c18d577cba572bfe4346b5cc7299c26" alt=""
data:image/s3,"s3://crabby-images/1c20a/1c20a7d4252c78266a34afb672793eb8992a393b" alt=""
data:image/s3,"s3://crabby-images/f1d7f/f1d7f358b2037c79fd189981b533fb6dce89987b" alt=""
结果:
data:image/s3,"s3://crabby-images/479f9/479f9c6cc7d3078670d55c23c5391943e3ee7424" alt=""
对于上面这个QueryWrapper,我们其实都没有用到这个EduTeacher,所以我们这样写也行:
data:image/s3,"s3://crabby-images/a02cb/a02cbe24536da411189f7a73c30af0b8210d9689" alt=""
data:image/s3,"s3://crabby-images/5e6e6/5e6e6b0d7048b4fb7fe70ca7f19eab7d4d7eac62" alt=""
data:image/s3,"s3://crabby-images/5bf9a/5bf9aeec09da418efa7793467cf86943d6786304" alt=""
但是这样,我们不能用链式编程了:
data:image/s3,"s3://crabby-images/2c810/2c810ab9e1bf01da640580b6af8dabfd31c26df9" alt=""
但是我们可以这样写:
data:image/s3,"s3://crabby-images/44e8c/44e8c9602b2217da52d4616b4c301603d8d6d23d" alt=""
执行结果:
data:image/s3,"s3://crabby-images/12fb6/12fb63d01de5fdac576f2fe489a2a59075fc3043" alt=""
其实相当于下面这样的sql执行:
data:image/s3,"s3://crabby-images/2af0c/2af0ce0e0fe67a4611b907914e35ebc44666b081" alt=""
当然哈,你也可以自己新建一个实体类,然后用这个新建的实体类接收查询的结果。但是你要是嫌麻烦的话,还是用上面这个List<Map<String,Object>>来接收吧。
例子1:
data:image/s3,"s3://crabby-images/95c99/95c99939eb17cd64f076e49ed877beb72200093f" alt=""
data:image/s3,"s3://crabby-images/b87af/b87af644af28277b4248173492e418a27fb847d6" alt=""
data:image/s3,"s3://crabby-images/f7496/f7496aad6c347f21cd6eb562f662d29f30e265f5" alt=""
data:image/s3,"s3://crabby-images/4d3b3/4d3b37c5e182567f0217533a0a3847f869315693" alt=""
data:image/s3,"s3://crabby-images/d7298/d72983e4421a4ed066b9fa36eb70d05d1953a496" alt=""
结果:
data:image/s3,"s3://crabby-images/74099/7409965005dd6fbc1346ba21e841bc9325e7cbcd" alt=""
4、自定义查询加上分页
其实就是先写好分页插件的配置,然后dao层接口方法的形参上面写一个IPage的参数,返回值改为IPage,到时候调用的时候把设置好参数后的IPage传过来就可以了,不用在dao层的方法上面写什么其他东西的。
具体如下:
data:image/s3,"s3://crabby-images/83ddc/83ddc27505e83a31d6abb359964826dda4a7e6fe" alt=""
data:image/s3,"s3://crabby-images/38337/38337e6aad0cca32e8c9b69296914cf324e5c6f2" alt=""
data:image/s3,"s3://crabby-images/b2845/b2845e80ea10b9a84e007014e5b9e7e178e578ed" alt=""
data:image/s3,"s3://crabby-images/111ca/111cad2595b92e0b23d6b55a5bd5c6bcba8f4e53" alt=""
data:image/s3,"s3://crabby-images/65a49/65a49ebd7537725bf1e33e5650f06a285cb87ad2" alt=""
结果:
data:image/s3,"s3://crabby-images/7fa3a/7fa3a30aa5a0d522295c47632ab788daaec5bd8a" alt=""
要给自定义的多表查询加上分页也一样哈,就写好分页插件的配置,然后dao层接口分页方法的形参上面写一个IPage的参数,返回值改为IPage,到时候调用的时候把设置好参数后的IPage传过来就可以了,不用在dao层的方法上面写什么其他东西。
例子如下:
data:image/s3,"s3://crabby-images/cb9fb/cb9fb4787521388796a9ba0749ac26697fc632b1" alt=""
data:image/s3,"s3://crabby-images/8d2a7/8d2a7cea1735985e34d771b79d202fe84ee222d4" alt=""
data:image/s3,"s3://crabby-images/f3786/f3786a09a142e516fce175c43f249e5584215a4d" alt=""
data:image/s3,"s3://crabby-images/9118d/9118d266e5ca8c7b698944e87ff8c1f1522a9887" alt=""
结果:
data:image/s3,"s3://crabby-images/98629/9862908082383a2d2545966fb477dea8bd530fe2" alt=""
数据查询结果不是封装到一个实体类里面的,而是放在一个Map集合里面的,那么我们得怎么加上分页呢?也是和前面一样的,就是接口的那个方法里面的第一个参数写的IPage中泛型改为Map<String,Object>就行了。
例子:
data:image/s3,"s3://crabby-images/bb569/bb56958ec4a67dcc8595eef3a5b87ade4a8f9ac8" alt=""
data:image/s3,"s3://crabby-images/5c7be/5c7bedecdeac950bfe53a6eed52c65af06f3810e" alt=""
data:image/s3,"s3://crabby-images/ad669/ad669c41bbde78c423b885ef063bbb4a7c6be5c7" alt=""
data:image/s3,"s3://crabby-images/4edb6/4edb668ca6ed1edccb6d147d18d786d5ce512470" alt=""
结果:
data:image/s3,"s3://crabby-images/a747f/a747f2a685aad51aa67c2f61b58f5ab7ddf1476b" alt=""
5、使用第三方工具做多表查询
上面我们做多表查询都是要自己写sql的,还是比较麻烦的,下面介绍一种不用自己写sql的方式来完成多表查询。
这个第三方工具是一个大佬封装的一个jar包,即mybatis-plus-join架包,这个架包可以支持MyBatis-Plus的多表联查。
官网如下:https://mybatisplusjoin.com/
或者你看这个博主写的也行:https://blog.csdn.net/weixin_39555954/article/details/128217887
快速入门的使用
一、引依赖
注意:要求mybatis plus version >= 3.4.0
xml
<dependency>
<groupId>com.github.yulichang</groupId>
<artifactId>mybatis-plus-join</artifactId>
<version>1.2.4</version>
</dependency>
二、使用方法
mapper继承MPJBaseMapper (必选)
service继承MPJBaseService (可选)
serviceImpl继承MPJBaseServiceImpl (可选)
三、(实战)多表查询
java
MPJLambdaWrapper<Map> mpjLambdaWrapper = new MPJLambdaWrapper();
mpjLambdaWrapper.select(ChatRecord::getId,ChatRecord::getRedMoney)
.select(OfShopMembers::getUsablePoint)
.select(ChatMultiList::getName)
.leftJoin(OfShopMembers.class,OfShopMembers::getId,ChatRecord::getId)
.leftJoin(ChatMultiList.class,ChatMultiList::getId,ChatRecord::getMultiId)
.eq(ChatRecord::getMemberId,3213);
List list = chatRecordMybatisJoinMapper.selectJoinList(Map.class, mpjLambdaWrapper);
对应查询语句
sql
SELECT
t.id,
t.red_money,
t1.username,
t2.name
FROM
chat_record t
LEFT JOIN of_shop_members t1 ON (t1.id = t.id)
LEFT JOIN chat_multi_list t2 ON (t2.id = t.multi_id)
WHERE
(t.member_id = 3213)
参数说明
1、select:表示查询的指定字段,一个select只能查一个表的
2、leftJoin:
第一个参数: 参与连表的实体类class
第二个参数: 连表的ON字段,这个属性必须是第一个参数实体类的属性
第三个参数: 参与连表的ON的另一个实体类属性
3、默认主表别名是t,其他的表别名以先后调用的顺序使用t1,t2,t3...
四、(实战)多表分页查询
java
MPJLambdaWrapper<Map> mpjLambdaWrapper = new MPJLambdaWrapper();
mpjLambdaWrapper.select(ChatRecord::getId,ChatRecord::getRedMoney)
.select(OfShopMembers::getUsablePoint)
.select(ChatMultiList::getName)
.leftJoin(OfShopMembers.class,OfShopMembers::getId,ChatRecord::getId)
.leftJoin(ChatMultiList.class,ChatMultiList::getId,ChatRecord::getMultiId)
.eq(ChatRecord::getMemberId,3213)
.orderByDesc(ChatRecord::getAddTime);
Page page = new Page(1,2);
IPage<Map> mapIPage = chatRecordMybatisJoinMapper.selectJoinPage(page, Map.class, mpjLambdaWrapper);
对应查询语句
sql
SELECT
t.id,
t.red_money,
t1.usable_point,
t2.name
FROM
chat_record t
LEFT JOIN of_shop_members t1 ON (t1.id = t.id)
LEFT JOIN chat_multi_list t2 ON (t2.id = t.multi_id)
WHERE
(t.member_id = 3213)
ORDER BY
t.add_time
DESC
LIMIT 2
我的测试:
一、引入依赖
这里我们测试的项目使用的是3.4.1的mybatis-plus-boot-starter
data:image/s3,"s3://crabby-images/742ab/742ab24dcc5858b9656fe161364c6ed9cd2dd73a" alt=""
我们进去看看里面指定的mybatis-plus的版本:
data:image/s3,"s3://crabby-images/d379b/d379ba54dafbf380670c01f32f83b5c3d9dcc55b" alt=""
看到是3.4.1版本的mybatis-plus,所以可以放心引入依赖了。
引入好后如下:
data:image/s3,"s3://crabby-images/d2695/d2695d77c00a1fb4d2561da4f6429fbf08e11809" alt=""
二、建表
我们新建三个表用来演示:
这三个表如下:
data:image/s3,"s3://crabby-images/6362c/6362cae4a7938a94f66bdef79f3ec43058128b08" alt=""
data:image/s3,"s3://crabby-images/48a6c/48a6c6b5e2b48137414cb3dee8a7479d1db21997" alt=""
data:image/s3,"s3://crabby-images/05aa4/05aa43fb9a597b3c255e54c1a40ad84ff4ad9f64" alt=""
三、新建实体类
data:image/s3,"s3://crabby-images/ecac7/ecac712ac07ea7e09da0452363912dfa0bb8f00b" alt=""
data:image/s3,"s3://crabby-images/92cd9/92cd9c50fe9f6a5999323421ef8b3e06eee3e18a" alt=""
data:image/s3,"s3://crabby-images/1ab67/1ab676ef29b361f458d6a13c1af7907ebd6bd4b4" alt=""
data:image/s3,"s3://crabby-images/008a0/008a0a4ef8715a676ed4f122471184eec5c70b86" alt=""
这个DTO里面写你要多表查询的全部数据。
四、新建dao层的接口
data:image/s3,"s3://crabby-images/38dc5/38dc5e528d26deb40ebd3502a134783f3add9a95" alt=""
data:image/s3,"s3://crabby-images/0644d/0644dac98651a8030de759dfd60da193495e283f" alt=""
data:image/s3,"s3://crabby-images/d1451/d14512ecfa9f3cedb46303966c69ac573c7fa54a" alt=""
五、我们直接测试
测试一:多表查询
你可以把下面的测试代码当作service层的代码。
data:image/s3,"s3://crabby-images/89b8e/89b8e570e6ed2f014340eb628be4aa94851267f5" alt=""
结果:
data:image/s3,"s3://crabby-images/a9200/a92007ac92844562b1006d2041b58eecfdbb3db2" alt=""
我们修改一下查询条件,看看结果:
data:image/s3,"s3://crabby-images/a717d/a717d115b51d15dd435319e6dd03701cbff398c9" alt=""
结果:
data:image/s3,"s3://crabby-images/d1449/d144953c0bf87bc12f6dbc42e5bfbdaaad2ee4db" alt=""
测试二:多表分页查询
data:image/s3,"s3://crabby-images/1398b/1398b014e07d5761f8cc67a4c792797fe0c68dc5" alt=""
结果:
data:image/s3,"s3://crabby-images/6f33c/6f33cb6ca29b350426f2103b432197ed49360cca" alt=""
数据库我们执行sql的结果如下(下面的这个sql执行的时候是没有带分页的limit的):
data:image/s3,"s3://crabby-images/edcfe/edcfe8f950b0b9997c2d7d5c5705a272c3028811" alt=""
看到,两个一组,第一组里面就是id为4和id为5的数据。所以测试完全正确。
但是要注意:这里之前是配置了分页插件的,要是没有配置分页插件,上面的执行结果不会有分页效果的。
data:image/s3,"s3://crabby-images/c21ab/c21ab7650766cb5e3e95dc9188cc19b21b9f9305" alt=""
测试三:多表分页查询且自定义别名
data:image/s3,"s3://crabby-images/a7687/a768747f59c9afaf2181099a20bd9b3fe3d1fa5a" alt=""
结果:
data:image/s3,"s3://crabby-images/49d5e/49d5e06f849034a1a22d09de4acac64129a20f9e" alt=""
测试四:多表查询分页且不把数据封装到实体类里面
data:image/s3,"s3://crabby-images/a19e5/a19e58d627960b09cd01282ac62d1a0a45958296" alt=""
结果:
data:image/s3,"s3://crabby-images/85690/85690fdd557a3f3deb05380ecae4ebe155862111" alt=""
测试五:多表分页查询,不封装到实体类,自定义别名
data:image/s3,"s3://crabby-images/5654a/5654a74e6faad8f1daec00981de5f27305730daf" alt=""
结果:
data:image/s3,"s3://crabby-images/024d2/024d29afb203c1b7c556792b7ac1f64ba0159f6f" alt=""