mybatis使用collection嵌套查詢的實現(xiàn)
在開發(fā)中,可能會遇到一對多的關系,這個時候,一條sql語句就難以勝任這個任務了。只能先執(zhí)行一條sql,然后根據(jù)返回的結(jié)果,再做一次sql關聯(lián)查詢,這個時候,使用mybatis的collection就可以實現(xiàn)。
如果第一次查詢返回的是一個list集合,那么,后續(xù)的查詢就是一個for循環(huán)。所以不使用collection的做法,在java語言中,就要分兩次查詢。一般而言,我們的列表查詢都是分頁查詢,所以集合數(shù)據(jù)不會太大,第二次for循環(huán)查詢效率還好。
下面介紹mybatis使用collection嵌套查詢解決這個問題。這里為了簡單,以員工與部門的關系來做這個實驗,其實員工與部門的關系還應該使用一個中間表來關聯(lián),這里只用兩張表。
表結(jié)構與數(shù)據(jù)如下所示:

數(shù)據(jù):

這里采用maven構建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>定義的實體:
員工信息:
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);
}
}運行單元測試,打印信息如下:

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

