经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 数据库/运维 » MyBatis » 查看文章
Mybatis自定义SQL的关系映射、分页、排序功能的实现
来源:jb51  时间:2021/1/18 19:54:47  对本文有异议

目的: 记录数据库表与实体对象之间不同的映射关系如何用mybatis的自定义sql和结果返回集处理。

1、三种对象映射关系

1.1 一对一

一个人对应一个身份证,一位同学对应一个班级,每个房间都有自己的房间号,当一个事物它对应另一个事物是唯一的,那么它们之间的关系就是一对一的。

这里我演示的案例是,一个学生有着一位老师

老师基础信息:

在这里插入图片描述

学生详细信息:

在这里插入图片描述

如果说,我们需要将两个表一起查出来,我们可以这么做:

问题: 如果对象的列重复了,必须要使用到别名

1、先定义实体结构,也就是我们返结果的实体类

  1. public class Student {
  2. @TableId
  3. private int id;
  4. private String name;
  5. private int tid;
  6. @TableField(exist = false)
  7. private Teacher teacher;
  8. }

Teacher:

  1. public class Teacher {
  2. @TableId
  3. private int id;
  4. private String name;
  5. }

2、 编写xml文件

这里有两种方式,使用association时的关键在于告诉mybatis如何加载关联(assocition)。

  • 嵌套查询:通过执行另外一个 SQL 映射语句来返回预期的复杂类型。
  • 嵌套结果:使用嵌套结果映射来处理重复的联合结果的子集。

第一种: 使用嵌套查询,也就是使用另一个sql

  1. // teacherMapper.xml
  2. <?xml version="1.0" encoding="UTF-8" ?>
  3. <!DOCTYPE mapper
  4. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  5. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  6. <mapper namespace="com.lll.mybatisplusdemo.mapper.TeacherMapper">
  7. <select id="getTeacher" parameterType="int" resultType="teacher">
  8. select * from teacher where id = #{id};
  9. </select>
  10. </mapper>
  11.  
  12. // studentMapper.xml
  13. <?xml version="1.0" encoding="UTF-8" ?>
  14. <!DOCTYPE mapper
  15. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  16. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  17.  
  18. <select id="getStudent2" parameterType="int" resultMap="getStudentMap2">
  19. select * from student where id =#{id};
  20. </select>
  21.  
  22. <resultMap id="getStudentMap2" type="student">
  23. <id column="id" property="id"></id>
  24. <result column="name" property="name"></result>
  25. <result column="tid" property="tid"></result>
  26. <association property="teacher" javaType="Teacher" column="tid" select="com.lll.mybatisplusdemo.mapper.TeacherMapper.getTeacher">
  27. <id column="id" property="id"></id>
  28. <result column="name" property="name"></result>
  29. </association>
  30. </resultMap>
  31. </mapper>

嵌套查询的方式很简单,但是对于大型数据集合和列表将不会表现很好。问题就是我们熟知的
“N+1 查询问题”。概括地讲, N+1 查询问题可以是这样引起的:

  • 你执行了一个单独的 SQL 语句来获取结果列表(就是“+1”)。
  • 对返回的每条记录,你执行了一个查询语句来为每个加载细节(就是“N”)。

第二种: 使用嵌套结果来映射联合查询来的数据

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.lll.mybatisplusdemo.mapper.StudentMapper">
  6. <select id="getStudent" parameterType="int" resultMap="getStudentMap">
  7. SELECT a.*,b.id as cid,b.name as cname FROM `student` as a,teacher as b WHERE a.id =#{id} and a.tid = b.id;
  8. </select>
  9. <resultMap id="getStudentMap" type="student">
  10. <id column="id" property="id"></id>
  11. <result column="name" property="name"></result>
  12. <result column="tid" property="tid"></result>
  13. <association property="teacher" javaType="Teacher">
  14. <id column="cid" property="id"></id>
  15. <result column="cname" property="name"></result>
  16. </association>
  17. </resultMap>
  18. </mapper>

我们在相应的mapper中添加方法接口便可以使用了。

1.2 一对多

案例:一个老师有多个学生

1、实体类

  1. public class Teacher {
  2. @TableId
  3. private int id;
  4. private String name;
  5. @TableField(exist = false)
  6. private List<Student> students;
  7. }

2、编写xml

同样还是,我们先来个嵌套结果映射

嵌套结果:

  1. // teacherMapper.xml
  2. <select id="getStudent" parameterType="int" resultMap="getStudentMap">
  3. SELECT a.*,b.id as cid,b.name as cname,b.tid
  4. from teacher as a , student as b
  5. where b.tid = a.id and a.id =#{id};
  6. </select>
  7.  
  8. <resultMap id="getStudentMap" type="teacher">
  9. <id column="id" property="id"></id>
  10. <result column="name" property="name"></result>
  11. <collection property="students" ofType="Student">
  12. <id column="cid" property="id"></id>
  13. <result column="cname" property="name"></result>
  14. <result column="tid" property="tid" ></result>
  15. </collection>
  16. </resultMap>

嵌套查询:

  1. // teacherMapper.xml
  2. <select id="getStudent2" parameterType="int" resultMap="getStudentMap2">
  3. select * from teacher as a where a.id = #{id}
  4.  
  5. </select>
  6. <resultMap id="getStudentMap2" type="teacher">
  7. <id column="id" property="id"></id>
  8. <result column="name" property="name"></result>
  9. <collection property="students" column="id" ofType="Student" select="com.lll.mybatisplusdemo.mapper.StudentMapper.getStudentByTid">
  10. </collection>
  11. </resultMap>
  12.  
  13. // studentMapper.xml
  14. <select id="getStudentByTid" parameterType="int" resultType="student">
  15. select * from student as a where a.tid = #{id}
  16. </select>

1.3 多对多

学生与课程是多对多的关系,与上面的一对多的操作方式是类似的

2、自定义sql如何做分页

mapper定义方法,方法传入page参数

  1. public interface UserMapper{
  2. IPage<User> selectPageVo(Page<User> page);
  3. }

userMapper.xml文件编写一个普通的返回结果是list的方法,mybatis会自动帮你做分页

  1. <select id="selectPage" resultType="com.baomidou.cloud.entity.UserVo">
  2. SELECT * FROM user
  3. </select>

3、自定义sql如何做排序

结论:使用order by,记住要使用'$',不能使用'#'

  1. <select id="selectPage" resultType="com.baomidou.cloud.entity.UserVo">
  2. SELECT * FROM user order by ${sortColumn} ${sortOrder}
  3. </select>

4、自定义sql中的#{}和${}的区别

1、传入的参数在SQL中显示不同

#传入的参数在SQL中显示为字符串(当成一个字符串),会对自动传入的数据加一个双引号。

例:使用以下SQL

  1. select id,name,age from student where id =#{id}

当我们传递的参数id为 “1” 时,上述 sql 的解析为:

  1. select id,name,age from student where id ="1"

$传入的参数在SqL中直接显示为传入的值

例:使用以下SQL

  1. select id,name,age from student where id =${id}

当我们传递的参数id为 “1” 时,上述 sql 的解析为:

  1. select id,name,age from student where id =1

2、#可以防止SQL注入的风险(语句的拼接);但$无法防止Sql注入。

3、$方式一般用于传入数据库对象,例如传入表名。

4、大多数情况下还是经常使用#,一般能用#的就别用$;但有些情况下必须使用$,例:MyBatis排序时使用order by 动态参数时需要注意,用$而不是#。

到此这篇关于Mybatis自定义SQL的关系映射、分页、排序的文章就介绍到这了,更多相关Mybatis自定义SQL的关系映射内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号