详解Mybatis之自动映射 自定义映射问题

news/2025/3/14 16:37:31/

编译软件:IntelliJ IDEA 2019.2.4 x64
操作系统:win10 x64 位 家庭版
Maven版本:apache-maven-3.6.3
Mybatis版本:3.5.6

文章目录

  • 一、Mybatis中的自动映射是什么?
  • 二、Mybatis中的自定义映射是什么?
  • 三、为什么要使用自定义映射[resultMap]?
  • 四、自定义映射[resultMap]可以适用哪些场景?
    • 4.1 resultMap之级联映射
      • 4.1.1 级联映射之association映射[1:1]
      • 4.1.2 级联映射之collection映射[1:m]
    • 4.2 总结ResultMap中的相关标签及属性
    • 4.3 分步查询
      • 4.3.1 一对一的关联关系
      • 4.3.2 一对多的关联关系
      • 4.3.3 扩展
  • 五、Mybatis如何使用延迟加载【懒加载】?


在这里插入图片描述


一、Mybatis中的自动映射是什么?

Mybatis中的自动映射不是什么高大上的技术名词,而是我们使用Mybatis框架进行持久化层开发时常用select元素中的常见属性resultType它可以自动将数据库内表中的字段与类中的属性进行关联映射,故而得名。


二、Mybatis中的自定义映射是什么?

👉定义

自定义映射,简而言之,就是resultMap。Mybatis官方将resultMap称为结果映射,在为一些比如连接的复杂语句编写映射代码的时候,一份 resultMap 能够代替实现同等功能的数千行代码。

👉设计思想

对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。


三、为什么要使用自定义映射[resultMap]?

💡原因

它可以解决自动映射[resultType]解决不了的两类问题

哪两类问题?

  • 🍓多表连接查询时,需要返回多张表的结果集

    不信?请看如下测试案例

    测试案例:通过员工id获取员工信息及员工所属的部门信息

    ①准备数据

    在这里插入图片描述

    在这里插入图片描述

    ②在Mapper接口书写相应的方法

    代码示例如下:

    //通过员工id获取员工信息及员工所属的部门信息
    public List<Employee> showempByempId(int empId);
    

    ③在接口对应的映射文件中书写相应的sql

    代码示例如下:

    <select id="showempByempId" resultType="employee">SELECTe.`id`,e.`last_name`,e.`email`,e.`salary`,d.`dpt_id`,d.`dpt_name`FROM`tbl_employee` e, `tbl_department` dWHEREe.`dept_id`=d.`dpt_id`ANDe.`id`=1;
    </select>
    

    ③测试

    @Test
    public void test01(){try {String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);//通过SqlSessionFactory对象调用openSession();SqlSession sqlSession = sqlSessionFactory.openSession();//获取EmployeeMapper的代理对象EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);List<Employee> employees = employeeMapper.showempByempId(1);System.out.println(employees);} catch (IOException e) {e.printStackTrace();}}
    

    在这里插入图片描述

    🙋 为什么员工所属的部门信息查不出来?

    🙇 原因

    书写的sql涉及到多表查询,映射文件中相应select子标签的属性为resultType。该属性不支持映射多表查询后的结果集,需要用到自定义映射来解决该问题

  • 🍓单表查询时,不支持驼峰式自动映射【如果不想为字段定义别名】

自定义映射【resultMap】:自动映射解决不了的问题,可以交给自定义映射

👇 注意

resultType与resultMap只能同时使用一个


四、自定义映射[resultMap]可以适用哪些场景?

4.1 resultMap之级联映射

🙋 何为级联映射?

👇 答曰

级联映射是指在保存主对象时,将关联的对象也一起保存到数据库中。例如,对于一对多或者多对一、多对多等关系对象时,当保存某个一对象时,与这个依赖的对象都应该自动保存或更新。比如:部门和员工表,一对多关系,当保存部门数据时,和部门有关联的员工表也同时保存。

用法案例

基于第三节中的案例,在映射文件中使用resultMap来解决多表查询后结果集中dept值为null的问题

代码示例如下:

①在在映射文件中使用自定义映射

<resultMap id="empAnddeptResultMap" type="employee"><!--  column:返回的结果集中的字段  property:返回值类型(employee)中的属性,要映射的类      --><!--  id属性是定义主键字段与属性之间的关联关系      --><id column="id" property="id"></id><!--  result属性是定义非主键字段与属性之间的关联关系      --><result column="last_name" property="lastName"></result><result column="email" property="email"></result><result column="salary" property="salary"></result><result column="dpt_id" property="dept.deptId"></result><result column="dpt_name" property="dept.deptName"></result>
</resultMap><select id="showempByempId" resultMap="empAnddeptResultMap">SELECTe.`id`,e.`last_name`,e.`email`,e.`salary`,d.`dpt_id`,d.`dpt_name`FROM`tbl_employee` e, `tbl_department` dWHEREe.`dept_id`=d.`dpt_id`ANDe.`id`=1;
</select>

②测试运行

在这里插入图片描述

4.1.1 级联映射之association映射[1:1]

👉特点

解决一对一的关联关系

👉用法案例

基于4.1小结中的案例,对映射文件中的sql部分进行association映射的改写,观察效果

代码示例如下:

①对映射文件中的sql部分进行association映射的改写

<resultMap id="empAnddeptResultMap" type="employee"><!--  column:返回的结果集中的字段  property:返回值类型(employee)中的属性,要映射的类      --><!--  id属性是定义主键字段与属性之间的关联关系      --><id column="id" property="id"></id><!--  result属性是定义非主键字段与属性之间的关联关系      --><result column="last_name" property="lastName"></result><result column="email" property="email"></result><result column="salary" property="salary"></result><!--  javaType: 用来指定某个属性(dept)或字段在 Java 代码中所对应的具体数据类型 (mybatis.pojo.Dept)    --><!--  dept属性指的是employee对象中的属性    --><association property="dept" javaType="mybatis.pojo.Dept"><id column="dpt_id" property="deptId"></id><result column="dpt_name" property="deptName"></result></association></resultMap><select id="showempByempId" resultMap="empAnddeptResultMap">SELECTe.`id`,e.`last_name`,e.`email`,e.`salary`,d.`dpt_id`,d.`dpt_name`FROM`tbl_employee` e, `tbl_department` dWHEREe.`dept_id`=d.`dpt_id`ANDe.`id`=1;
</select>

②测试运行

在这里插入图片描述

4.1.2 级联映射之collection映射[1:m]

👉特点

解决一对多的关联关系

👉用法案例

根据部门编号查询对应的部门信息,然后拿着部门编号去员工表里去找所属的员工信息(此时部门与员工是一对多的关系)

代码示例如下:

准备数据

在这里插入图片描述

①在DeptMapper接口书写相应的方法

//根据部门编号查询对应的部门信息,然后拿着部门编号去员工表里去找所属的员工信息(一对多)
public Dept showEmployeesByDeptId(int deptId);

②在在DeptMapper接口对应的映射文件中书写相应的sql

<!-- collection property="employees" ofType="mybatis.pojo.Employee" -> 在Dept类中名为employees的集合中存储的元素类型  --><resultMap id="showEmployeesByDeptIdResultMap" type="dept"><id property="deptId" column="dpt_id"></id><result property="deptName" column="dpt_name"></result><collection property="employees" ofType="mybatis.pojo.Employee"><!--  id属性是定义主键字段与属性之间的关联关系      --><id column="id" property="id"></id><!--  result属性是定义非主键字段与属性之间的关联关系      --><result column="last_name" property="lastName"></result><result column="email" property="email"></result><result column="salary" property="salary"></result></collection></resultMap><select id="showEmployeesByDeptId" resultMap="showEmployeesByDeptIdResultMap">SELECTe.`id`,e.`last_name`,e.`email`,e.`salary`,d.`dpt_id`,d.`dpt_name`FROM`tbl_employee` e, `tbl_department` dWHEREe.`dept_id`=d.`dpt_id`ANDd.`dpt_id`=#{dptId};</select>

③测试

@Test
public void test04(){try {String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);//通过SqlSessionFactory对象调用openSession();SqlSession sqlSession = sqlSessionFactory.openSession();//获取EmployeeMapper的代理对象DeptMapper deptMapper = sqlSession.getMapper(DeptMapper.class);//查询部门编号为1的部门信息,并得到所属员工的所有员工信息Dept dept = deptMapper.showEmployeesByDeptId(1);System.out.println(dept);} catch (IOException e) {e.printStackTrace();}}

在这里插入图片描述

4.2 总结ResultMap中的相关标签及属性

  • resultMap标签自定义映射标签

    • id属性定义唯一标识
    • type属性设置映射类型
  • resultMap子标签

    • id标签定义主键字段与属性关联关系

    • result标签定义非主键字段与属性关联关系

      • column属性定义表中字段名称
      • property属性定义类中属性名称
    • associationi标签定义一对一的关联关系

      • property属性定义关联关系属性
      • javaType属性定义关联关系属性的类型
      • select属性设置分步查询SQL全路径
      • colunm属性设置分步查询SQL中需要参数
      • fetchType:设置局部延迟加载【懒加载】
    • collection标签定义一对多的关联关系

      • property属性定义关联关系属性
      • ofType属性定义关联关系属性类型
      • select属性设置分步查询SQL全路径
      • colunm属性设置分步查询SQL中需要参数
      • fetchType:设置局部延迟加载【懒加载】是否开启

4.3 分步查询

🙋 为什么使用分步查询【分步查询优势】?

将多表连接查询,改为【分步单表查询】,从而提高程序运行效率

4.3.1 一对一的关联关系

👉用法案例

使用分步查询实现通过员工id获取员工信息及员工所属的部门信息,比如说1.通过员工id获取员工信息,2.通过员工信息中的部门id获得所属部门得信息(员工与部门是一对一的关系,即一个员工只能归属一个部门)

代码示例如下:

①在EmployeeMapper接口中定义实现通过员工id获取员工信息的方法

//使用分步查询实现通过员工id获取员工信息及员工所属的部门信息
//1.通过员工id获取员工信息
//2.通过员工信息中的部门id获得所属部门得信息
public Employee selectEmpByempId(int empId);

②在DeptMapper接口中定义实现通过从查出来的员工信息中的部门编号去查所属部门信息

//通过部门id查询所属部门得信息
public Dept selectDeptByDeptId(int deptId);

③在EmployeeMapper接口对应的映射文件书写相应的sql

 <resultMap id="selectEmpByempIdResultMap" type="mybatis.pojo.Employee"><id property="id" column="id"></id><result property="lastName" column="last_name"></result><result property="email" column="email"></result><result property="salary" column="salary"></result><!--  column="deptId"  设置分步查询SQL中需要得参数dept_Id;将此值传入到mybatis.mapper.DeptMapper中的selectDeptByDeptId()方法中     --><association property="dept"select="mybatis.mapper.DeptMapper.selectDeptByDeptId"column="dept_Id"   ></association></resultMap><select id="selectEmpByempId" resultMap="selectEmpByempIdResultMap">SELECT`id`,`last_name`,`email`,`salary`,`dept_id`FROM`tbl_employee`WHERE`id`=#{empId};</select>

④在DeptMapper接口对应的映射文件中书写相应的sql

<resultMap id="selectDeptByDeptIdResultMap" type="dept"><id property="deptId" column="dpt_Id"></id><result property="deptName" column="dpt_name"></result>
</resultMap><select id="selectDeptByDeptId" resultMap="selectDeptByDeptIdResultMap">selectdpt_Id,dpt_namefromtbl_departmentwheredpt_Id=#{dptId}
</select>

⑤测试

@Test
public void test02(){try {String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);//通过SqlSessionFactory对象调用openSession();SqlSession sqlSession = sqlSessionFactory.openSession();//获取EmployeeMapper的代理对象EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);Employee employee = employeeMapper.selectEmpByempId(2);System.out.println(employee);} catch (IOException e) {e.printStackTrace();}}

在这里插入图片描述

4.3.2 一对多的关联关系

👉用法案例

通过部门id获取部门信息,及部门所属员工信息【使用分步查询来实现】,其中按1.通过部门id获取部门信息;2.通过部门id获取员工信息等这两个步骤完成分步查询

代码示例如下:

①在DeptMapper接口中书写查询通过部门id获取部门信息的方法

//通过部门id获取部门信息
public Dept showEmployeesByDeptIdBetter(int deptId);

②在DeptMapper接口对应的映射文件中书写相应的sql

<!--  type="dept" 设置映射类型为dept,为什么不是dept类的全称,因为我在配置文件给它起了别名  --><resultMap id="showEmployeesByDeptIdBetterResultMap" type="dept"><id property="deptId" column="dpt_id"></id><result property="deptName" column="dpt_name"></result><collection property="employees"select="mybatis.mapper.EmployeeMapper#selectEmployeeByempId"column="dpt_Id"></collection></resultMap><select id="showEmployeesByDeptIdBetter" resultMap="showEmployeesByDeptIdBetterResultMap">SELECT`dpt_id`,`dpt_name`FROM`tbl_department`WHERE`dpt_id`=#{dptId};</select>

③在EmployeeMapper接口中书写查询通过部门id获取所属员工信息的方法

//通过部门id查询对应的员工信息
public Employee selectEmployeeByempId(int empId);

④在EmployeeMapper接口对应的映射文件中书写相应的sql

<select id="showEmployeesByDeptIdBetter" resultMap="showEmployeesByDeptIdBetterResultMap">SELECT`dpt_id`,`dpt_name`FROM`tbl_department`WHERE`dpt_id`=#{dptId};</select>

⑤测试

@Test
//测试分步查询版(根据部门编号查询对应的部门信息,然后拿着部门编号去员工表里去找所属的员工信息)
public void test05(){try {String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);//通过SqlSessionFactory对象调用openSession();SqlSession sqlSession = sqlSessionFactory.openSession();//获取EmployeeMapper的代理对象DeptMapper deptMapper = sqlSession.getMapper(DeptMapper.class);Dept dept = deptMapper.showEmployeesByDeptId(1);System.out.println(dept);} catch (IOException e) {e.printStackTrace();}}

在这里插入图片描述

4.3.3 扩展

如果使用分步查询时,需要传递给调用的查询中多个参数,则需要将多个参数封装成 Map来进行传递,语法如下:{k1=v1,k2=v2}


五、Mybatis如何使用延迟加载【懒加载】?

🙋什么是延迟加载?

需要时加载,不需要暂时不加载,如何理解?举个生活中的例子,就好比当你非常饥饿时,才会去吃饭

👉优势

可以提高程序运行效率

👉语法

🚀①全局设置

在核心配置文件中这样写,示例代码如下

<settings><!--    开启驼峰命名自动映射    --><setting name="mapUnderscoreToCamelCase" value="true"/><!--  开启全局延迟加载模式      --><setting name="lazyLoadingEnabled" value="true"/><!--  关闭按需延迟加载模式,在3.4.2版本及以后该步骤可省略      --><setting name="aggressiveLazyLoading" value="false"/>
</settings>

👉用法案例

在核心配置文件开启全局延迟加载模式,借助8.6小结中的案例代码,演示全局延迟加载模式的效果

代码示例如下:

测试运行如下

try {String resource = "mybatis-config.xml";InputStream inputStream = Resources.getResourceAsStream(resource);SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);//通过SqlSessionFactory对象调用openSession();SqlSession sqlSession = sqlSessionFactory.openSession();//获取EmployeeMapper的代理对象EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);Employee employee = employeeMapper.selectEmpByempId(2);System.out.println("employee.getLastName() = "+employee.getLastName());System.out.println("--------------------------------------");System.out.println("employee.getDept() = "+employee.getDept());} catch (IOException e) {e.printStackTrace();
}

在这里插入图片描述

🚀②局部设置

  • fetchType
    • eager关闭局部延迟加载
    • lazy开启局部延迟加载

👉用法案例

在上述案例中EmployeeMapper接口对应的映射文件里属性id的值为"selectEmpByempId"的sql设置关闭延迟加载(已经设置了全局延迟加载模式,这里再开启局部延迟加载,效果不明显,遂采用关闭局部延迟加载测试效果)

代码示例如下:

    <resultMap id="selectEmpByempIdResultMap" type="mybatis.pojo.Employee"><id property="id" column="id"></id><result property="lastName" column="last_name"></result><result property="email" column="email"></result><result property="salary" column="salary"></result><!--  column="deptId"  设置分步查询SQL中需要得参数dept_Id;将此值传入到mybatis.mapper.DeptMapper中的selectDeptByDeptId()方法中     --><!--   fetchType="lazy" 为此方法设置局部延迟加载                   --><association property="dept"select="mybatis.mapper.DeptMapper.selectDeptByDeptId"column="dept_Id"fetchType="lazy"   ></association></resultMap><select id="selectEmpByempId" resultMap="selectEmpByempIdResultMap">SELECT`id`,`last_name`,`email`,`salary`,`dept_id`FROM`tbl_employee`WHERE`id`=#{empId};</select>

运行测试如下

在这里插入图片描述



http://www.ppmy.cn/news/980330.html

相关文章

认识雪花id

首先,个人理解,雪花id不是全球的,它只能保证一个分布式服务的范围内的ID是不重复的. 一.SnowFlake 雪花算法 SnowFlake 中文意思为雪花&#xff0c;故称为雪花算法。最早是 Twitter 公司在其内部用于分布式环境下生成唯一 ID。在2014年开源 scala 语言版本。 雪花算法的原理…

二、Java框架基础02 XML

二、XML 2.1 XML 简介 XML 即可扩展标记语言&#xff0c;一种简单的数据存储语言&#xff0c;使用一系列简单的标记来描述结构化数据 XML 的特点 XML 与操作系统&#xff0c;编程语言的开发平台无关规范统一&#xff0c;实现不同系统之间的数据交互 2.1.1 XML 的文档结构 以下…

Rust vs Go:常用语法对比(四)

题图来自 Go vs. Rust performance comparison: The basics 61. Get current date 获取当前时间 package mainimport ( "fmt" "time")func main() { d : time.Now() fmt.Println("Now is", d) // The Playground has a special sandbox, so you …

Linux:Linux的发展史和作用有哪些?

文章目录 Linux是什么&#xff1f;Linux的开源特征为什么要学习Linux&#xff1f;Linux的应用场景有哪些&#xff1f; Linux是什么&#xff1f; 简单来说&#xff0c;Linux就是操作系统&#xff0c;它和Windows等软件一样&#xff0c;都只是操作系统&#xff0c;并无区别 Linu…

DAY2,Qt(继续完善登录框,信号与槽的使用 )

1.继续完善登录框&#xff0c;当登录成功时&#xff0c;关闭登录界面&#xff0c;跳转到新的界面中&#xff0c;来回切换页面&#xff1b; ---mychat.h chatroom.h---两个页面头文件 #ifndef MYCHAT_H #define MYCHAT_H#include <QWidget> #include <QDebug> /…

SpringBoot整合JavaMail

SpringBoot整合JavaMail 简单使用-发送简单邮件 介绍协议 导入坐标 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mail</artifactId></dependency>添加配置 spring:mail:host: smtp.qq.co…

国密SSL优势及应用场景

国密SSL的优势主要有以下几点&#xff1a; 更高的安全性&#xff1a;国密算法采用的是国家密码管理局推荐的算法&#xff0c;相对于传统的SSL协议更加安全。 更好的性能&#xff1a;国密算法是国家密码管理局推荐的算法&#xff0c;其加密效率与密钥长度相比传统算法更高。 更…

前端开发面试题

vue的父子组件传值用的是什么? 通过 props 传值。 Vue的生命周期方法有哪些? beforeCreate 在实例初始化之后,数据观测(data observe)和 event/watcher 事件配置之前被调用。在当前阶段 data、methods、computed 以及 watch 上的数据和方法都不能被访问。created 实例已…