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

本文详细介绍了Mybatis中的自动映射与自定义映射(resultMap)的概念,解释了为何需要使用自定义映射,以及在多表查询和单表查询中的应用场景,包括级联映射中的association和collection。此外,还探讨了分步查询的优势和使用方法,并讲解了Mybatis的延迟加载机制及其配置。
摘要由CSDN通过智能技术生成

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


在这里插入图片描述


一、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">
        SELECT
            e.`id`,
            e.`last_name`,
            e.`email`,
            e.`salary`,
            d.`dpt_id`,
            d.`dpt_name`
        FROM
            `tbl_employee` e, `tbl_department` d
        WHERE
            e.`dept_id`=d.`dpt_id`
        AND
            e.`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">
    SELECT
        e.`id`,
        e.`last_name`,
        e.`email`,
        e.`salary`,
        d.`dpt_id`,
        d.`dpt_name`
    FROM
        `tbl_employee` e, `tbl_department` d
    WHERE
        e.`dept_id`=d.`dpt_id`
    AND
        e.`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">
    SELECT
        e.`id`,
        e.`last_name`,
        e.`email`,
        e.`salary`,
        d.`dpt_id`,
        d.`dpt_name`
    FROM
        `tbl_employee` e, `tbl_department` d
    WHERE
        e.`dept_id`=d.`dpt_id`
    AND
        e.`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">
        SELECT
            e.`id`,
            e.`last_name`,
            e.`email`,
            e.`salary`,
            d.`dpt_id`,
            d.`dpt_name`
        FROM
            `tbl_employee` e, `tbl_department` d
        WHERE
            e.`dept_id`=d.`dpt_id`
        AND
            d.`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">
    select
        dpt_Id,
        dpt_name
    from
        tbl_department
    where
        dpt_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>

运行测试如下

在这里插入图片描述


  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
MyBatis是一个开源的持久层框架,它可以将数据库操作与Java对象之间的映射关系进行配置,提供了一种简单且灵活的方式来访问数据库。 在MyBatis中,映射文件是用于定义SQL语句和结果映射的配置文件。下面是映射文件的详解: 1. 命名空间(namespace):映射文件中的命名空间用于对SQL语句进行分类和管理,可以通过命名空间来引用映射文件中定义的SQL语句。 2. 结果映射(resultMap):结果映射用于将查询结果映射到Java对象上。可以通过resultMap标签定义结果映射,指定查询结果与Java对象之间的对应关系。 3. SQL语句(select、insert、update、delete):映射文件中可以定义各种SQL语句,包括查询、插入、更新和删除等操作。可以通过标签(如select、insert、update、delete)来定义SQL语句,并在其中编写具体的SQL语句。 4. 参数映射(parameterType):参数映射用于将Java对象作为参数传递给SQL语句。可以通过parameterType属性指定参数类型,也可以通过parameterMap标签定义参数映射。 5. 返回值映射(resultType):返回值映射用于将查询结果转换为Java对象。可以通过resultType属性指定返回值类型,也可以通过resultMap标签引用已定义的结果映射。 6. 动态SQL:MyBatis支持动态SQL,可以根据条件动态生成SQL语句。可以使用if、choose、when、otherwise等标签来实现动态SQL的编写。 7. 参数传递:映射文件中可以使用#{}或者${}来传递参数。#{}会将参数转义后传递给数据库,${}会直接将参数拼接到SQL语句中。 8. 引用其他映射文件:映射文件可以通过include标签引用其他映射文件,可以将一些通用的SQL语句定义在公共的映射文件中,然后在需要的地方引用。 以上是对MyBatis映射文件的详解,希望能对你有所帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陌上少年,且听这风吟

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值