mybatis使用collection嵌套查詢的實(shí)現(xiàn)
在開(kāi)發(fā)中,可能會(huì)遇到一對(duì)多的關(guān)系,這個(gè)時(shí)候,一條sql語(yǔ)句就難以勝任這個(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語(yǔ)言中,就要分兩次查詢。一般而言,我們的列表查詢都是分頁(yè)查詢,所以集合數(shù)據(jù)不會(huì)太大,第二次for循環(huán)查詢效率還好。
下面介紹mybatis使用collection嵌套查詢解決這個(gè)問(wèn)題。這里為了簡(jiǎn)單,以員工與部門的關(guān)系來(lái)做這個(gè)實(shí)驗(yàn),其實(shí)員工與部門的關(guān)系還應(yīng)該使用一個(gè)中間表來(lái)關(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>測(cè)試類:
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)行單元測(cè)試,打印信息如下:

這個(gè)程序,在第一次根據(jù)名稱查詢員工信息之后,返回id=1,name=aa,age=18,接著,根據(jù)id=1 查詢了部門表,其實(shí)這里的id=1,作為參數(shù)傳入部門表中,就成了user_id對(duì)應(yīng)的參數(shù),然后就查詢出了兩個(gè)部門記錄id=1,id=3,最后打印的員工信息里面,depts就是一個(gè)集合。
使用這個(gè)嵌套查詢,需要注意的是collection有如下屬性:
property 實(shí)體中對(duì)應(yīng)的屬性,這里是depts。 javaType 實(shí)體屬性的類型,這里是一個(gè)集合,所以使用java.util.ArrayList表示。 ofType 集合范型中的類型,這里是部門信息,對(duì)應(yīng)java類Department select 嵌套子查詢的ID column 這里最關(guān)鍵,也比較難理解,默認(rèn)一個(gè)參數(shù),可以直接寫(xiě)column = "id",最后根據(jù)參數(shù)類型匹配。這里其實(shí)是傳入子查詢中的參數(shù),也就是子查詢的關(guān)聯(lián)屬性u(píng)ser_id對(duì)應(yīng)的參數(shù)值,在collection這里就是主sql中查詢出來(lái)的列值,如果這里id有了別名,比如emp_id,這里就應(yīng)該寫(xiě)column = "emp_id"。還有一種寫(xiě)法,通過(guò)大括號(hào)來(lái)表示,這種寫(xiě)法可以傳入多個(gè)參數(shù)(多個(gè)參數(shù)用逗號(hào)隔開(kāi))。對(duì)于本示例而言,正確的寫(xiě)法就是column={userId=id},userId對(duì)應(yīng) queryByUserId查詢語(yǔ)句中 SELECT * FROM xx_dept WHERE user_id = #{userId} 參數(shù)userId。id就是主sql查詢出來(lái)的列id值,就是xx_emp對(duì)應(yīng)的id列值。
我個(gè)人在使用collection的時(shí)候,大概明白這種嵌套查詢的作用,但是就是不明白,怎么傳遞參數(shù),后來(lái)看過(guò)一些例子,原來(lái)主要點(diǎn)就在column屬性這里。
到此這篇關(guān)于mybatis使用collection嵌套查詢的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)mybatis collection嵌套查詢內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot RocketMq實(shí)現(xiàn)過(guò)程詳解
這篇文章主要介紹了Springboot RocketMq實(shí)現(xiàn)過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05
SpringSecurity rememberme功能實(shí)現(xiàn)過(guò)程解析
這篇文章主要介紹了SpringSecurity rememberme功能實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
Springboot pom項(xiàng)目間接依賴包版本與預(yù)期不符原因解決分析
這篇文章主要介紹了Springboot pom項(xiàng)目間接依賴包版本與預(yù)期不符原因解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08
Java 自旋鎖(spinlock)相關(guān)知識(shí)總結(jié)
這篇文章主要介紹了Java 自旋鎖(spinlock)相關(guān)知識(shí)總結(jié),幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2021-02-02
Java中的遞歸詳解(用遞歸實(shí)現(xiàn)99乘法表來(lái)講解)
這篇文章主要介紹了Java中的遞歸詳解(用遞歸實(shí)現(xiàn)99乘法表來(lái)講解),本文給出了普通的99乘法實(shí)現(xiàn)方法和用遞歸實(shí)現(xiàn)的方法,并對(duì)比它們的不同,體現(xiàn)出遞歸的運(yùn)用及理解,需要的朋友可以參考下2015-03-03
Java線程的start方法回調(diào)run方法的操作技巧
面試過(guò)程中經(jīng)常會(huì)被面試官問(wèn)到為什么我們調(diào)用start()方法時(shí)會(huì)執(zhí)行run()方法,為什么不能直接調(diào)用run()方法,問(wèn)的一頭霧水,今天小編給大家介紹下Java線程的start方法回調(diào)run方法的操作技巧,需要的朋友參考下吧2017-11-11
SpringBoot與knife4j的整合使用過(guò)程
Knife4j?是一個(gè)基于Swagger構(gòu)建的開(kāi)源?JavaAPI文檔工具,主要包括兩大核心功能:文檔說(shuō)明和在線調(diào)試,這篇文章主要介紹了SpringBoot與knife4j的整合使用,需要的朋友可以參考下2024-08-08
IDEA中l(wèi)og4j 無(wú)法輸出到本地 properties配置無(wú)效問(wèn)題
這篇文章主要介紹了IDEA中l(wèi)og4j 無(wú)法輸出到本地 properties配置無(wú)效問(wèn)題,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-10-10

