詳解Mybatis中javaType和ofType的區(qū)別
一. 背景描述
今天,壹哥給學(xué)生講解了Mybatis框架,學(xué)習(xí)了基礎(chǔ)的ORM框架操作及多對(duì)一的查詢。在練習(xí)的時(shí)候,小張同學(xué)突然舉手求助,說(shuō)在做預(yù)習(xí)作業(yè)使用一對(duì)多查詢時(shí),遇到了ReflectionException 異常 。
二. 情景再現(xiàn)
1. 實(shí)體類
為了給大家講清楚這個(gè)異常的產(chǎn)生原因,壹哥先列出今天案例中涉及到的兩張表:書籍表和書籍類型表。這兩張表中存在著簡(jiǎn)單的多對(duì)一關(guān)系,實(shí)體類如下:
@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映射文件
上課時(shí),壹哥講解的關(guān)聯(lián)查詢是通過查詢書籍信息,并同時(shí)對(duì)書籍類型查詢。即在查詢Book對(duì)象時(shí)i,同時(shí)查詢出BookType對(duì)象。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> <!-- 單個(gè)對(duì)象的關(guān)聯(lián),javaType是指實(shí)體類的類型--> <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"> <!-- 事務(wù)管理器--> <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. 測(cè)試代碼
接著我們對(duì)上面的配置進(jìn)行測(cè)試。
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(); }
學(xué)生按照我講的內(nèi)容,測(cè)試沒有問題。在后續(xù)的預(yù)習(xí)練習(xí)中,要求實(shí)現(xiàn)在BookType中添加List屬性books,在查詢BookType對(duì)象同時(shí)將該類型的Book對(duì)象集合查出。小張同學(xué)有了如下實(shí)現(xiàn)思路。
5. 修改實(shí)體類
@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. 編寫測(cè)試類
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(); }
然后就出現(xiàn)了一開始提到的異常:
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=很不錯(cuò)的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=很不錯(cuò)的Java書籍, createTime=2022-05-27, type=null, imgPath=174cc662fccc4a38b73ece6880d8c07e)' Cause: java.lang.IllegalArgumentException: argument type mismatch
三. 異常分析
上面的 異常提示 , 是 說(shuō)在 BookType類中的books屬性設(shè)置有問題 。 我們來(lái)仔細(xì)查看一下代碼,發(fā)現(xiàn)是因?yàn)橹苯?/strong> 復(fù)制了之前的關(guān)系配置, 在配置文件中 使用javaType 節(jié)點(diǎn) , 但正確的 應(yīng)該 是 使用ofType。如下圖所示:
四. 解析
那么為什么有的關(guān)系配置要使用javaType,而有的地方又要使用ofType呢?
這我們就不得不說(shuō)說(shuō)Mybatis的底層原理了!在關(guān)聯(lián)映射中,如果是單個(gè)的JavaBean對(duì)象,那么可以使用javaType;而如果是集合類型,則需要寫ofType。以下是Mybatis的官方文檔原文:
五. 結(jié)尾
雖然上面的代碼中只是因?yàn)橐粋€(gè)單詞的不同,卻造成了不小的錯(cuò)誤。我們的程序是嚴(yán)格的,小問題就可能會(huì)耽誤你很久的時(shí)間。就比如我們的小張同學(xué),在求助壹哥之前已經(jīng)找bug找了一個(gè)小時(shí)......最后壹哥一眼就給他看出了問題所在,他都無(wú)語(yǔ)凝噎了.....
現(xiàn)在你明白javaType和ofType用法上的區(qū)別了嗎?如果你還有其他什么問題,可以在評(píng)論區(qū)留言或私信哦!關(guān)注Java架構(gòu)棧,干貨天天都不斷。
?到此這篇關(guān)于詳解Mybatis中javaType和ofType的區(qū)別的文章就介紹到這了,更多相關(guān)Mybatis javaType和ofType內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot 改造成https訪問的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot 改造成https訪問的實(shí)現(xiàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2020-10-10Java實(shí)現(xiàn)可視化走迷宮小游戲的示例代碼
這篇文章主要介紹了Java如何實(shí)現(xiàn)可視化走迷宮小游戲。本程序適用于java程序員鞏固類與對(duì)象、文件讀取、事件響應(yīng)、awt包中各種工具的相關(guān)概念以及對(duì)邏輯能力的鍛煉,需要的可以參考一下2022-11-11SpringBoot配置文件中系統(tǒng)環(huán)境變量存在特殊字符的處理方式
這篇文章主要介紹了SpringBoot配置文件中系統(tǒng)環(huán)境變量存在特殊字符的處理方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02Java設(shè)計(jì)模式之備忘錄模式(Memento模式)介紹
這篇文章主要介紹了Java設(shè)計(jì)模式之備忘錄模式(Memento模式)介紹,memento是一個(gè)保存另外一個(gè)對(duì)象內(nèi)部狀態(tài)拷貝的對(duì)象,這樣以后就可以將該對(duì)象恢復(fù)到原先保存的狀態(tài),需要的朋友可以參考下2015-03-03