mybatis使用collection嵌套查詢的實(shí)現(xiàn)
在開發(fā)中,可能會遇到一對多的關(guān)系,這個(gè)時(shí)候,一條sql語句就難以勝任這個(gè)任務(wù)了。只能先執(zhí)行一條sql,然后根據(jù)返回的結(jié)果,再做一次sql關(guān)聯(lián)查詢,這個(gè)時(shí)候,使用mybatis的collection就可以實(shí)現(xiàn)。
如果第一次查詢返回的是一個(gè)list集合,那么,后續(xù)的查詢就是一個(gè)for循環(huán)。所以不使用collection的做法,在java語言中,就要分兩次查詢。一般而言,我們的列表查詢都是分頁查詢,所以集合數(shù)據(jù)不會太大,第二次for循環(huán)查詢效率還好。
下面介紹mybatis使用collection嵌套查詢解決這個(gè)問題。這里為了簡單,以員工與部門的關(guān)系來做這個(gè)實(shí)驗(yàn),其實(shí)員工與部門的關(guān)系還應(yīng)該使用一個(gè)中間表來關(guān)聯(lián),這里只用兩張表。
表結(jié)構(gòu)與數(shù)據(jù)如下所示:
數(shù)據(jù):
這里采用maven構(gòu)建springboot+mybatis-plus+mysql工程。
<dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.2.0</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency>
定義的實(shí)體:
員工信息:
package com.xxx.springboot.mybatis.domain; import java.util.List; import com.baomidou.mybatisplus.annotation.TableName; import lombok.Data; @Data @TableName("xx_emp") public class Employee { private Integer id ; private String name ; private int age ; private List<Department> depts ; }
部門信息
package com.xxx.springboot.mybatis.domain; import com.baomidou.mybatisplus.annotation.TableName; import lombok.Data; @Data @TableName("xx_dept") public class Department { private Integer id ; private String name ; private Integer userId ; }
Mapper接口
package com.xxx.springboot.mybatis.mapper; import org.apache.ibatis.annotations.Mapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.xxx.springboot.mybatis.domain.Department; import com.xxx.springboot.mybatis.domain.Employee; @Mapper public interface EmpMapper extends BaseMapper<Employee>{ Employee queryByName(String name); Department queryByUserId(Integer userId); }
EmpMapper.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.xxx.springboot.mybatis.mapper.EmpMapper"> <resultMap id="empMap" type="com.xxx.springboot.mybatis.domain.Employee" > <id column="id" property="id" /> <result column="name" property="name" /> <result column="age" property="age" /> <collection property="depts" javaType="java.util.ArrayList" ofType="com.xxx.springboot.mybatis.domain.Department" select="com.xxx.springboot.mybatis.mapper.EmpMapper.queryByUserId" column="{userId=id}"></collection> </resultMap> <resultMap id="deptMap" type="com.xxx.springboot.mybatis.domain.Department"> <id column="id" property="id"/> <result column="name" property="name"/> <result column="user_id" property="userId"/> </resultMap> <select id="queryByName" resultMap="empMap"> SELECT * FROM xx_emp WHERE name = #{name} </select> <select id="queryByUserId" resultMap="deptMap"> SELECT * FROM xx_dept WHERE user_id = #{userId} </select> </mapper>
測試類:
package com.xxx.springboot; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.xxx.springboot.mybatis.domain.Employee; import com.xxx.springboot.mybatis.mapper.EmpMapper; @RunWith(SpringRunner.class) @SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT) public class MyBatisTest { @Autowired private EmpMapper empMapper; @Test public void testEmpQuery() { Employee emp = empMapper.queryByName("aa"); System.out.println(emp); } }
運(yùn)行單元測試,打印信息如下:
這個(gè)程序,在第一次根據(jù)名稱查詢員工信息之后,返回id=1,name=aa,age=18,接著,根據(jù)id=1 查詢了部門表,其實(shí)這里的id=1,作為參數(shù)傳入部門表中,就成了user_id對應(yīng)的參數(shù),然后就查詢出了兩個(gè)部門記錄id=1,id=3,最后打印的員工信息里面,depts就是一個(gè)集合。
使用這個(gè)嵌套查詢,需要注意的是collection有如下屬性:
property 實(shí)體中對應(yīng)的屬性,這里是depts。 javaType 實(shí)體屬性的類型,這里是一個(gè)集合,所以使用java.util.ArrayList表示。 ofType 集合范型中的類型,這里是部門信息,對應(yīng)java類Department select 嵌套子查詢的ID column 這里最關(guān)鍵,也比較難理解,默認(rèn)一個(gè)參數(shù),可以直接寫column = "id",最后根據(jù)參數(shù)類型匹配。這里其實(shí)是傳入子查詢中的參數(shù),也就是子查詢的關(guān)聯(lián)屬性user_id對應(yīng)的參數(shù)值,在collection這里就是主sql中查詢出來的列值,如果這里id有了別名,比如emp_id,這里就應(yīng)該寫column = "emp_id"。還有一種寫法,通過大括號來表示,這種寫法可以傳入多個(gè)參數(shù)(多個(gè)參數(shù)用逗號隔開)。對于本示例而言,正確的寫法就是column={userId=id},userId對應(yīng) queryByUserId查詢語句中 SELECT * FROM xx_dept WHERE user_id = #{userId} 參數(shù)userId。id就是主sql查詢出來的列id值,就是xx_emp對應(yīng)的id列值。
我個(gè)人在使用collection的時(shí)候,大概明白這種嵌套查詢的作用,但是就是不明白,怎么傳遞參數(shù),后來看過一些例子,原來主要點(diǎn)就在column屬性這里。
到此這篇關(guān)于mybatis使用collection嵌套查詢的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)mybatis collection嵌套查詢內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot RocketMq實(shí)現(xiàn)過程詳解
這篇文章主要介紹了Springboot RocketMq實(shí)現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05SpringSecurity rememberme功能實(shí)現(xiàn)過程解析
這篇文章主要介紹了SpringSecurity rememberme功能實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Springboot pom項(xiàng)目間接依賴包版本與預(yù)期不符原因解決分析
這篇文章主要介紹了Springboot pom項(xiàng)目間接依賴包版本與預(yù)期不符原因解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08Java 自旋鎖(spinlock)相關(guān)知識總結(jié)
這篇文章主要介紹了Java 自旋鎖(spinlock)相關(guān)知識總結(jié),幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2021-02-02Java中的遞歸詳解(用遞歸實(shí)現(xiàn)99乘法表來講解)
這篇文章主要介紹了Java中的遞歸詳解(用遞歸實(shí)現(xiàn)99乘法表來講解),本文給出了普通的99乘法實(shí)現(xiàn)方法和用遞歸實(shí)現(xiàn)的方法,并對比它們的不同,體現(xiàn)出遞歸的運(yùn)用及理解,需要的朋友可以參考下2015-03-03Java線程的start方法回調(diào)run方法的操作技巧
面試過程中經(jīng)常會被面試官問到為什么我們調(diào)用start()方法時(shí)會執(zhí)行run()方法,為什么不能直接調(diào)用run()方法,問的一頭霧水,今天小編給大家介紹下Java線程的start方法回調(diào)run方法的操作技巧,需要的朋友參考下吧2017-11-11IDEA中l(wèi)og4j 無法輸出到本地 properties配置無效問題
這篇文章主要介紹了IDEA中l(wèi)og4j 無法輸出到本地 properties配置無效問題,非常不錯,具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-10-10