resultType
从这条语句中返回的期望类型的类的完全限定名或别名,注意如果是集合,那应该是集合可以包含的类型,而不能是集合本身。
下面我们来讨论其返回List和Map的情况:
(1)select返回List
在select元素中,如果返回的是一个集合,要在sql映射中写集合中元素的类型。
我们想实现通过名字来进行模糊查询,返回Employee类型的List集合。
public List getEmpsByLastNameLike(String lastName);
然后我们在sql映射文件中配置如下:
注意resultType
填入的是Employee
的类型而不是List
类型。
测试方法,查询名字带有e的:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static org.junit.Assert.*;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test9() throws IOException {
//获取SqlSessionFactory
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
//获取SqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
//获取接口的实现类对象:会为接口自动的创建一个代理对象,代理对象去执行增删改查
EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);
//调用方法
List list = employeeMapper.getEmpsByLastNameLike("%e%");
for (Employee employee : list){
System.out.println(employee);
}
}finally {
//关闭
sqlSession.close();
}
}
}
结果:
(2)select记录封装Map
有两种情况,一种是返回单个实体类的Map;另一种是返回多个实体类的Map。
(3)返回单个实体类的Map
如果返回的是单个实体类的Map,那么在select
元素中的resultType
的值就是map
。
我们想返回的Map是key为记录的列名,值为记录的值。我们首先在接口中定义一个方法:
public Map getEmpByIdReturnMap(Integer id);
然后在sql映射文件中给予配置,注意resultType
的值是map
:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static org.junit.Assert.*;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test8() throws IOException {
//获取SqlSessionFactory
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
//获取SqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
//获取接口的实现类对象:会为接口自动的创建一个代理对象,代理对象去执行增删改查
EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);
Map map = employeeMapper.getEmpByIdReturnMap(1);
System.out.println(map);
}finally {
//关闭
sqlSession.close();
}
}
}
(4)返回多个实体类的Map
我们想返回多个实体类的Map的话,那么resultType
的值应该是实体类,而不是map
本身。
我们想返回多个Employee组成的Map,key为Employee
中的id
顺序,在这里我们需要使用到@MapKey
注解来指明我们要封装哪个属性为Map的key,值是Employee
对象,我们先在接口中定义方法:
//告诉MyBatis封装这个Map的时候使用哪个属性作为Map的key
@MapKey("id")
public Map getEmpByLastNameReturnMap(String lastName);
SQL 映射文件:
测试方法:
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test9() throws IOException {
//获取SqlSessionFactory
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
//获取SqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
//获取接口的实现类对象:会为接口自动的创建一个代理对象,代理对象去执行增删改查
EmployeeMapper employeeMapper = sqlSession.getMapper(EmployeeMapper.class);
Map map = employeeMapper.getEmpByLastNameReturnMap("%e%");
System.out.println(map);
}finally {
//关闭
sqlSession.close();
}
}
}
结果:
resultMap
(1)通过resultMap
实现高级结果映射集
其属性如下:
-
type
:自定义规则的Java类型 -
id
:唯一id方便引用
上述代码中,
子标签指定主键列的封装规则。
-
column
:指定哪一列 -
property
:指定对应的JavaBean属性
子标签定义普通列的封装规则,属性与
的属性用法一致。对于主键来说也可以使用
定义规则,但是使用
定义主键会有底层优化,所以我们推荐主键使用
标签来定义
定义好规则后,
的id
是myEmp
,这个id
可供下面的标签引用,即
resultMap="myEmp"
。
测试方法如下:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import com.cerr.mybatis.dao.EmployeeMapperPlus;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static org.junit.Assert.*;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test10() throws IOException {
//获取SqlSessionFactory
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
//获取SqlSession
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
//获取接口的实现类对象:会为接口自动的创建一个代理对象,代理对象去执行增删改查
EmployeeMapperPlus mapper = sqlSession.getMapper(EmployeeMapperPlus.class);
Employee employee = mapper.getEmpById(1);
System.out.println(employee);
}finally {
//关闭
sqlSession.close();
}
}
(2)使用resultMap
进行关联查询
首先我们定义一个Department类并且新建一张表:
/**新建表**/
create table tbl_dept(
id int(11) primary key auto_increment,
dept_name varchar(255)
)
/**对tb1_employee表新增一列**/
alter table tb1_employee add column d_id int(11);
/**新增外键**/
alter table tb1_employee add constraint fk_emp_dept foreign key(d_id)
references tbl_dept(id);
package com.cerr.mybatis;
public class Department {
private Integer id;
private String departmentName;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getDepartmentName() {
return departmentName;
}
public void setDepartmentName(String departmentName) {
this.departmentName = departmentName;
}
@Override
public String toString() {
return "Department{" +
"id=" + id +
", departmentName='" + departmentName + '\'' +
'}';
}
}
(3)使用级联属性封装结果来进行关联查询
在EmployeeMapperPlus
接口中新增一个方法:
public Employee getEmpAndDept(Integer id);
SQL映射文件配置如下,在resultMap
中使用department.id
这种级联写法来封装结果集:
(4)使用association
定义关联对象封装规则
我们也可以使用
,
可以指定联合的JavaBean对象,其中的property
属性指定哪个属性是联合的对象,javaType
属性指定这个属性对象的类型。
示例如下:
(5)使用association
进行分步查询
对于上面的关联对象查询中,我们可以将其分解为两步,其SQL如下:
select * from tb1_employee where id = ?
select id,dept_name departmentName from tbl_dept where id = ?
即先通过id在tb1_employee
表中查出信息,并且该记录中有一个d_id
字段,我们将该字段作为第二条sql语句的参数去tbl_dept
表中查出记录,并把该记录封装成Department
,然后赋值给Employee
的department
属性。此时就可以用到我们的分步查询。
在EmployeeMapper
接口中新增一个方法:
public Employee getEmpByIdStep(Integer id);
我们新建一个DepartmentMapper
接口:
package com.cerr.mybatis.dao;
import com.cerr.mybatis.Department;
public interface DepartmentMapper {
public Department getDeptById(Integer id);
}
然后新建一个SQL映射文件DepartmentMapper.xml
:
上述配置使用了association
标签来配置分步查询的方法,select
属性表明当前属性是调用select
指定的方法查出的结果,要填入该方法的完整名字(包括全类名)。column
属性指定将哪一列的值传给这个方法。
测试方法:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import com.cerr.mybatis.dao.EmployeeMapperPlus;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static org.junit.Assert.*;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test12() throws IOException {
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
EmployeeMapperPlus mapper = sqlSession.getMapper(EmployeeMapperPlus.class);
Employee employee = mapper.getEmpByIdStep(1);
System.out.println(employee);
System.out.println(employee.getDepartment());
}finally {
//关闭
sqlSession.close();
}
}
}
结果如下:
延迟加载
我们上述方法查询的时候,每次查询Employee
对象的时候,都将Department一起查询出来了,而我们想将部门信息等到我们要使用的时候再去查询。只需要在上述的分步查询的基础上在全局配置文件中配置lazyLoadingEnabled
和aggressiveLazyLoading
即可。
我们修改上面分步查询中的测试方法,不使用到department
字段:
@Test
public void test12() throws IOException {
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
EmployeeMapperPlus mapper = sqlSession.getMapper(EmployeeMapperPlus.class);
Employee employee = mapper.getEmpByIdStep(1);
System.out.println(employee.getEmail());
}finally {
//关闭
sqlSession.close();
}
}
在控制台中看其SQL的信息(此处使用了log4j
,详情看这篇文章:MyBatis | 使用log4j在控制台输出SQL语句):
没有查询Department
表:
若需要使用到
department
字段时:
@Test
public void test12() throws IOException {
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
EmployeeMapperPlus mapper = sqlSession.getMapper(EmployeeMapperPlus.class);
Employee employee = mapper.getEmpByIdStep(1);
System.out.println(employee.getEmail());
System.out.println(employee.getDepartment());
}finally {
//关闭
sqlSession.close();
}
}
(6)使用collection定义关联的集合类型元素的封装规则
例如我们案例中,对于一个Department
类中,嵌套了一个Employee
类型的集合,那么我们想在查询Department
的时候顺便将其集合元素查询出来,我们就可以通过使用collection
嵌套结果集的方式,定义关联的集合类型元素的封装规则。
collection
标签:定义关联集合类型的属性的封装规则,其有两个比较重要的属性:
-
property
:定义关联集合类型的属性的封装规则 -
ofType
:指定集合里面元素的类型
我们先在Department
类中加入集合元素
private List emps;
再加入对应的getter/setter
,还有重写toString()
:
public List < Employee > getEmps() {
return emps;
}
public void setEmps(List < Employee > emps) {
this.emps = emps;
}
@Override
public String toString() {
return "Department{" +
"id=" + id +
", departmentName='" + departmentName + '\'' +
", emps=" + emps +
'}';
}
在DepartmentMapper
接口中新增方法:
public Department getDeptByIdPlus(Integer id);
SQL映射文件配置:
测试方法:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.DepartmentMapper;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import com.cerr.mybatis.dao.EmployeeMapperPlus;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test13() throws IOException {
SqlSessionFactory factory = getSqlSessionFactory();
SqlSession session = factory.openSession();
try{
DepartmentMapper mapper = session.getMapper(DepartmentMapper.class);
Department department = mapper.getDeptByIdPlus(1);
System.out.println(department);
System.out.println(department.getEmps());
}finally {
session.close();
}
}
}
结果:
(7)使用collection定义关联的集合元素分步查询
这个与association
进行分步查询的结果类似,只是association
是针对单个元素,而collection
针对集合元素。
对于上面使用collection定义关联的集合类型元素的封装规则中的代码中,我们可以拆分为:
select * from tbl_dept where id = ?;
select * from tb1_employee where d_id = ?;
既先查tbl_dept
表中的记录,然后根据Department
的id
字段的记录去tb1_employee
表中查询符合条件的Employee
记录,并封装成集合然后给Department
包装。
EmployeeMapperPlus
接口方法:
public List getEmpsByDeptId(Integer deptId);
SQL映射文件EmployeeMapperPlus.xml
:
DepartmentMapperPlus
接口方法:
public Department getDeptByIdStep(Integer id);
SQL映射文件:
测试方法:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.DepartmentMapper;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import com.cerr.mybatis.dao.EmployeeMapperPlus;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test13() throws IOException {
SqlSessionFactory factory = getSqlSessionFactory();
SqlSession session = factory.openSession();
try{
DepartmentMapper mapper = session.getMapper(DepartmentMapper.class);
Department department = mapper.getDeptByIdStep(1);
System.out.println(department);
System.out.println(department.getEmps());
}finally {
session.close();
}
}
}
结果:
collection的扩展
对于分步查询传递多列值的时候,可以将其值封装成map
再传递进column
属性。其格式如{key1=column1,key2=column2,...}
。
对于上面的配置文件,我们可以修改为column="{deptId=id}"
,具体配置如下:
在collection
中有一个fetchType
属性,其有两种取值:
-
lazy
:表示使用延迟加载 -
eager
:表示使用立即加载
(8)使用discriminator鉴别器
MyBatis可以使用discriminator
判断某列的值,然后根据某列的值改变其封装行为。
对于discriminator
标签,有如下两个属性
-
column
:指定判定的列名 -
javaType
:指定列值对应的java类型
discriminator
标签还有一个子标签case
,其属性如下:
-
value
:列的值 -
resultType
:指定封装的结果类型
现在我们有一个要求,如果在查询Employee
时,查出的是女生,就把部门信息查询出来,否则就不查询;如果是男生,就把last_name
这一列的值赋值给email
。
这样的话,我们首先要使用
标签进行自定义结果集封装,对于其他的列值,我们就依旧使用
和
标签来封装,对于gender
属性我们使用
来判定,SQL映射文件如下:
测试方法:
package com.cerr.mybatis;
import com.cerr.mybatis.dao.DepartmentMapper;
import com.cerr.mybatis.dao.EmployeeMapper;
import com.cerr.mybatis.dao.EmployeeMapperAnnotation;
import com.cerr.mybatis.dao.EmployeeMapperPlus;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
import org.junit.Test;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Map;
public class MyBatisTest {
//获取SQLSessionFactory
public SqlSessionFactory getSqlSessionFactory() throws IOException {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
return new SqlSessionFactoryBuilder().build(inputStream);
}
@Test
public void test12() throws IOException {
SqlSessionFactory sqlSessionFactory = getSqlSessionFactory();
SqlSession sqlSession = sqlSessionFactory.openSession();
try {
EmployeeMapperPlus mapper = sqlSession.getMapper(EmployeeMapperPlus.class);
Employee employee = mapper.getEmpByIdStep(4);
System.out.println(employee);
System.out.println(employee.getDepartment());
}finally {
//关闭
sqlSession.close();
}
}
}
上述代码中,id=4的Employee是女生,因此会查询Department:
我们将传入的id值改为1(男生),所以不会查询Department,所以获取其
department
值的时候是null
,并且email
的值与其lastName
的值一样: