首页 > 编程语言 >细节决定成败:探究Mybatis中javaType和ofType的区别

细节决定成败:探究Mybatis中javaType和ofType的区别

时间:2023-07-10 23:01:53浏览次数:43  
标签:name private javaType String book Mybatis type id ofType

一. 背景描述

今天给学生讲解了Mybatis框架,学习了基础的ORM框架操作及多对一的查询。在练习的时候,小张同学突然举手求助,说在做预习作业使用一对多查询时,遇到了ReflectionException 异常 

二. 情景再现

1. 实体类

为了给大家讲清楚这个异常的产生原因,壹哥先列出今天案例中涉及到的两张表:书籍表和书籍类型表。这两张表中存在着简单的多对一关系,实体类如下:

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Book {
    private Integer id;
    private String name;
    private String author;
    private String bookDesc;
    private String createTime;
    private BookType type;
    private String imgPath;
}

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class BookType {
    private Integer id;
    private String name;
}
复制代码

2.BookMapper.xml映射文件

上课时,讲解的关联查询是通过查询书籍信息,并同时对书籍类型查询。即在查询Book对象时i,同时查询出BookType对象。BookMapper.xml映射文件如下:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.qf.day7.dao.BookDAO">
    <resultMap id="booksMap" type="com.qf.day7.entity.Books">
        <id property="id" column="id"></id>
        <result property="name" column="name"></result>
        <result property="author" column="author"></result>
        <result property="bookDesc" column="book_desc"></result>
        <result property="createTime" column="create_time"></result>
        <result property="imgPath" column="img_path"></result>
        <!--        单个对象的关联,javaType是指实体类的类型-->
        <association property="type" javaType="com.qf.day7.entity.BookType">
            <id property="id" column="type_id"></id>
            <result property="name" column="type_name"></result>
        </association>
    </resultMap>
    
    <select id="findAll" resultMap="booksMap">
        SELECT
        b.id,
        b.`name`,
        b.author,
        b.book_desc,
        b.create_time,
        b.img_path,
        t.id type_id,
        t.`name` type_name
        FROM
        books AS b
        INNER JOIN book_type AS t ON b.type_id = t.id
    </select>
</mapper>
复制代码

3. 核心配置

核心配置文件如下:mybatisCfg.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <typeAliases>
        <package name="com.qf.day7.entity"/>
    </typeAliases>

    <environments default="development">
        <environment id="development">
<!--            事务管理器-->
            <transactionManager type="JDBC"></transactionManager>
<!--            使用mybatis自带连接池-->
            <dataSource type="POOLED">
<!--                jdbc四要素-->
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url"
                          value="jdbc:mysql://localhost:3306/books?useUnicode=true&characterEncoding=utf-8&useSSL=false"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </dataSource>
        </environment>
    </environments>

    <mappers>
        <mapper resource="mapper/BookMapper.xml"></mapper>
        <mapper resource="mapper/BookTypeMapper.xml"></mapper>
    </mappers>
</configuration>
复制代码

4. 测试代码

接着我们对上面的配置进行测试。

public class BookDAOTest {
    private SqlSessionFactory factory;
    
    @Before
    public void setUp() throws Exception {
        final InputStream inputStream = Resources.getResourceAsStream("mybatisCfg.xml");
        factory = new SqlSessionFactoryBuilder().build(inputStream);
    }

    @Test
    public void findAll() {
        final SqlSession session = factory.openSession();
        final BookDAO bookDAO = session.getMapper(BookDAO.class);
        final List<Book> list = bookDAO.findAll();
        list.stream().forEach(System.out::println);
        session.close();
    }
  }
复制代码

学生按照我讲的内容,测试没有问题。在后续的预习练习中,要求实现在BookType中添加List属性books,在查询BookType对象同时将该类型的Book对象集合查出。小张同学有了如下实现思路。

5. 修改实体类

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Book {
    private Integer id;
    private String name;
    private String author;
    private String bookDesc;
    private String createTime;
    private String imgPath;
}

@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class BookType {
    private Integer id;
    private String name;
    private List<Book> books;
}
复制代码

6. 添加映射文件BookTypeMapper.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.qf.day7.dao.BookTypeDAO">
    <resultMap id="bookTypeMap" type="com.qf.day7.entity.BookType">
        <id column="id" property="id"></id>
        <result column="name" property="name"></result>
        <collection property="books" javaType="com.qf.day7.entity.Book">
            <id property="id" column="book_id"></id>
            <result property="name" column="book_name"></result>
            <result property="author" column="author"></result>
            <result property="bookDesc" column="book_desc"></result>
            <result property="createTime" column="create_time"></result>
            <result property="imgPath" column="img_path"></result>
        </collection>
    </resultMap>

    <select id="findById" resultMap="bookTypeMap">
SELECT
            b.id book_id,
            b.`name` book_name,
            b.author,
            b.book_desc,
            b.create_time,
            b.img_path,
            t.id,
            t.`name`
        FROM
            books AS b
        INNER JOIN book_type AS t ON b.type_id = t.id
        where t.id = #{typeId}
    </select>
</mapper>
复制代码

7. 编写测试类

public class BookTypeDAOTest {
    private SqlSessionFactory factory;
    @Before
    public void setUp() throws Exception {
        final InputStream inputStream = Resources.getResourceAsStream("mybatisCfg.xml");
        factory = new SqlSessionFactoryBuilder().build(inputStream);
    }

    @Test
    public void findById() {
        final SqlSession session = factory.openSession();
        final BookTypeDAO bookTypeDAO = session.getMapper(BookTypeDAO.class);
        BookType bookType = bookTypeDAO.findById(1);
        for (Book book : bookType.getBooks()) {
            System.out.println(book.getName());
        }
        session.close();
    }
复制代码

然后就出现了一开始提到的异常:

org.apache.ibatis.exceptions.PersistenceException: 
### Error querying database.  Cause: org.apache.ibatis.reflection.ReflectionException: Could not set property 'books' of 'class com.qf.day7.entity.BookType' with value 'Book(id=1, name=Java从入门到精通, author=千锋, bookDesc=很不错的Java书籍, createTime=2022-05-27, type=null, imgPath=174cc662fccc4a38b73ece6880d8c07e)' Cause: java.lang.IllegalArgumentException: argument type mismatch
### The error may exist in mapper/BookTypeMapper.xml
### The error may involve com.qf.day7.dao.BookTypeDAO.findById
### The error occurred while handling results
### SQL: SELECT             b.id book_id,             b.`name` book_name,             b.author,             b.book_desc,             b.create_time,             b.img_path,             t.id,             t.`name`         FROM             books AS b         INNER JOIN book_type AS t ON b.type_id = t.id         where t.id = ?
### Cause: org.apache.ibatis.reflection.ReflectionException: Could not set property 'books' of 'class com.qf.day7.entity.BookType' with value 'Book(id=1, name=Java从入门到精通, author=千锋, bookDesc=很不错的Java书籍, createTime=2022-05-27, type=null, imgPath=174cc662fccc4a38b73ece6880d8c07e)' Cause: java.lang.IllegalArgumentException: argument type mismatch
复制代码

三. 异常分析

上面的 异常提示   说在 BookType类中的books属性设置有问题  我们来仔细查看一下代码,发现是因为直接 复制了之前的关系配置, 在配置文件中 使用javaType 节点  但正确的 应该  使用ofType。如下图所示:

细节决定成败:探究Mybatis中javaType和ofType的区别_xml

四. 解析

那么为什么有的关系配置要使用javaType,而有的地方又要使用ofType呢?

这我们就不得不说说Mybatis的底层原理了!在关联映射中,如果是单个的JavaBean对象,那么可以使用javaType;而如果是集合类型,则需要写ofType。以下是Mybatis的官方文档原文:

细节决定成败:探究Mybatis中javaType和ofType的区别_xml_02

五. 结尾

虽然上面的代码中只是因为一个单词的不同,却造成了不小的错误。我们的程序是严格的,小问题就可能会耽误你很久的时间。就比如我们的小张同学,在求助之前已经找bug找了一个小时......最后一眼就给他看出了问题所在,他都无语凝噎了.....

现在你明白javaType和ofType用法上的区别了吗?如果你还有其他什么问题,可以在评论区留言或私信哦!关注我,干货天天都不断。

标签:name,private,javaType,String,book,Mybatis,type,id,ofType
From: https://blog.51cto.com/u_16173760/6681775

相关文章

  • SpringBoot+Mybatis搭建之采坑记录(持续更新...)
    Stoppingservice[Tomcat] 1.缺少Serivce注解无法启动tomcat 2.包名错误3.写了注解没写参数使用Eclipse调试Springboot项目时总是直接进入SilentExitExceptionHandler解决方案:Window-->Preference-->java-->debug-->Suspendexecutiononuncaughtexceptions选项前面的勾......
  • Eclipse搭建springBoot进阶篇-SpringBoot+Mybatis
    Eclipse搭建springBoot入门篇 在入门篇了搭建了一个简单的SpringBoot的Demo,在进阶篇将真正引入实战,也就是能支撑基本的企业中小型项目开发。本次通过springBoot+Mybatis+maven等搭建一个适用于开发中小型项目的简单架构pom文件需要更新maven的jar,对java包进行下载。<?xmlversi......
  • 1-MyBatisPlus 入门案例与简介
    1.入门案例MybatisPlus(简称MP)是基于MyBatis框架基础上开发的增强型工具,旨在简化开发、提供效率。开发方式基于MyBatis使用MyBatisPlus基于Spring使用MyBatisPlus基于SpringBoot使用MyBatisPlusSpringBoot整合MybatisPlus具体实现步骤为:创建数据库......
  • spring-boot集成mybatis-plus
    spring-boot集成mybatis-plus目录spring-boot集成mybatis-plus依赖配置application.properties文件启动类添加@Mapper注解编码实体类Mapper类Service类测试@Test测试测试结果依赖 <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter<......
  • mybatis-八股文
    mybatis的优缺点-※优点:1.SQL写在XML里面,与业务代码分离,因此相对比较灵活,便于统一管理2.spring集成很方便,由于使用的JDBC连接数据库,因此,不需要针对不同数据专门做兼容缺点:SQL的编写,SQL的语法都强关联数据库的类型,因此,无法随意切换数据库mybatis中#{}和${}的区别?#的......
  • day116-mybatis的逆向工程
    mybatis的逆向工程正向工程:先创建Java实体类,由框架负责根据实体类生成数据库表。逆向工程:先创建数据库表,由框架负责根据数据库表,反向生成如下资源:Java实体类,mapper接口,mapper映射文件步骤添加依赖与插件<dependencies><!--Mybatis核心--><dependency>......
  • 几步带你掌握MyBatis Plus
    Mybatis-Plus(简称MP)是一款Mybatis的增强工具,它是在Mybatis的基础上实现的简化开发工具。Mybatis-Plus给我们提供了开箱即用的CRUD操作、自动生成代码、注解SQL的编写、分页等常用功能,可以减少我们的代码量,增加开发效率。接下来我就来带大家快速使用Mybatis-Plus,Let’sGo!My......
  • mybatis: 正确使用mybatis中的mapperLocations配置多个xml扫描路径
    <!--myBatis文件--><beanid="sqlSessionFactory"class="org.mybatis.spring.SqlSessionFactoryBean"><propertyname="dataSource"ref="dataSource"/><!--自动扫描entity目录,省掉Configu......
  • 71. mybatis 如何获取插入的id【从零开始学习SpirngBoot】
      【从零开始学习SpirngBoot—常见异常汇总】      在之前的文章已经讲过springboot集成mybatis了,但是忘记说一个很重要的知识点了,那就是获取获取主键id,这篇文章补充下,springboot集成mybatis看之前文章:       其实这个也很简单,主要是使用@Options注解,核心代......
  • 其实MyBatis的插件机制可以帮我们解决工作很多问题,建议收藏
    MyBatis插件插件是一种常见的扩展方式,大多数开源框架也都支持用户通过添加自定义插件的方式来扩展或者改变原有的功能,MyBatis中也提供的有插件,虽然叫插件,但是实际上是通过拦截器(Interceptor)实现的,在MyBatis的插件模块中涉及到责任链模式和JDK动态代理,这两种设计模式的技术知识也......