有时侯,我们用SQL取得的结果需要映射到类似Map<key, Bean>这样的数据结构中或是映射到多个实体类中时,我们就需要使用到resultMap。下面用3个例子说明Mybatis高级结果映射的用法。

<环境准备>

请参照上一篇blog<Mybatis Guide>

<数据准备>

创建班级--学生表的一对多的数据结构

 --创建班级表
CREATE TABLE tempdb..test_class
(
class_id int NOT NULL PRIMARY KEY,
class_name varchar(255),
class_admin_name varchar(255)
);
--创建学生表
CREATE TABLE tempdb..test_student
(
student_id int NOT NULL PRIMARY KEY,
class_id int,
student_name varchar(255),
student_age int,
student_address varchar(255)
);
--班级表中插入数据
insert tempdb..test_class values (101, 'Class 1, Grade 1', 'Mr. Zhang');
insert tempdb..test_class values (102, 'Class 2, Grade 1', 'Mr. Li');
insert tempdb..test_class values (103, 'Class 3, Grade 1', 'Mr. Wang');
insert tempdb..test_class values (201, 'Class 1, Grade 2', 'Mr. Zhao');
insert tempdb..test_class values (202, 'Class 2, Grade 2', 'Mr. Liu');
--学生表中插入数据
insert tempdb..test_student values (101001, 101, 'Name 1', 20, 'Address 1');
insert tempdb..test_student values (101002, 101, 'Name 2', 21, 'Address 2');
insert tempdb..test_student values (101003, 101, 'Name 3', 20, 'Address 3');
insert tempdb..test_student values (102001, 102, 'Name 4', 22, 'Address 4');
insert tempdb..test_student values (102002, 102, 'Name 5', 21, 'Address 5');
insert tempdb..test_student values (201001, 201, 'Name 6', 23, 'Address 6');
insert tempdb..test_student values (201002, 201, 'Name 7', 22, 'Address 7');
insert tempdb..test_student values (202001, 202, 'Name 8', 22, 'Address 8');

<创建实体类Entity>

 package mybatistest;

 import java.io.Serializable;

 public class Clazz implements Serializable {

     private static final long serialVersionUID = 1L;

     private Integer classId;

     private String className;

     private String classAdminName;

     public Integer getClassId() {
return classId;
} public void setClassId(Integer classId) {
this.classId = classId;
} public String getClassName() {
return className;
} public void setClassName(String className) {
this.className = className;
} public String getClassAdminName() {
return classAdminName;
} public void setClassAdminName(String classAdminName) {
this.classAdminName = classAdminName;
} @Override
public String toString() {
return "Clazz [classId=" + classId + ", className=" + className + ", classAdminName=" + classAdminName + "]";
}
}

Class

 package mybatistest;

 import java.io.Serializable;

 public class Student implements Serializable {

     private static final long serialVersionUID = 1L;

     private Integer studentId;

     private Integer classId;

     private String studentName;

     private Integer studentAge;

     private String studentAddress;

     public Integer getStudentId() {
return studentId;
} public void setStudentId(Integer studentId) {
this.studentId = studentId;
} public String getStudentName() {
return studentName;
} public Integer getClassId() {
return classId;
} public void setClassId(Integer classId) {
this.classId = classId;
} public void setStudentName(String studentName) {
this.studentName = studentName;
} public Integer getStudentAge() {
return studentAge;
} public void setStudentAge(Integer studentAge) {
this.studentAge = studentAge;
} public String getStudentAddress() {
return studentAddress;
} public void setStudentAddress(String studentAddress) {
this.studentAddress = studentAddress;
} @Override
public String toString() {
return "Student [classId=" + classId + ", studentId=" + studentId + ", studentName=" + studentName
+ ", studentAge=" + studentAge + ", studentAddress=" + studentAddress + "]";
}
}

Student

<创建StudentsPerClass(OutputBean)>

创建一个Class的OutBean,其中包含classId,Class实体类,及Student实体类List以对应班级--学生的1对多关系。

 package mybatistest;

 import java.io.Serializable;
import java.util.List; public class StudentsPerClass implements Serializable { private static final long serialVersionUID = 1L; private Integer classId; private Clazz clazz; private List<Student> students; public Integer getClassId() {
return classId;
} public void setClassId(Integer classId) {
this.classId = classId;
} public Clazz getClazz() {
return clazz;
} public void setClazz(Clazz clazz) {
this.clazz = clazz;
} public List<Student> getStudents() {
return students;
} public void setStudents(List<Student> students) {
this.students = students;
} @Override
public String toString() {
StringBuilder sb = new StringBuilder(
"StudentsPerClass [classId=" + classId + ", clazz=" + clazz + ", students=" + System.lineSeparator());
students.forEach(p -> {
sb.append("--").append(p).append(System.lineSeparator());
});
return sb.append("]").toString();
}
}

StudentsPerClass

<创建三个SQL实例>

(1) 用Map<学号, 学生实体类>映射一个学生集合

(2) 用List<StudentsPerClass>映射班级--学生的1对多关系

(3) 用Map<班级号, StudentsPerClass>映射班级--学生集合

ClassStudentRepository.java

 package mybatistest;

 import java.util.List;
import java.util.Map; import org.apache.ibatis.annotations.MapKey; public interface ClassStudentRepository { // 获取一个班级的所有学生[Map<[Student Id], [Student Info]>]
@MapKey("studentId")
Map<Integer, Student> getStudentsFromOneClass(Integer classId); // 获取一个年级的所有学生
List<StudentsPerClass> getStudentsFromOneGrade(Integer gradeId); // 获取所有学生
@MapKey("classId")
Map<Integer, StudentsPerClass> getAllStudents();
}

ClassStudentRepository.xml

注意黄色部分非常重要,只有设置了1对多关系中的主键class_id,Mybatis才能自动映射这种1对多的嵌套关系

 <?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="mybatistest.ClassStudentRepository">
<!-- // 获取一个班级的所有学生[Map<[Student Id], [Student Info]>] -->
<select id="getStudentsFromOneClass" resultMap="studentsFromOneClassMap">
SELECT
t2.student_id,
t2.class_id,
t2.student_name,
t2.student_age,
t2.student_address
FROM
tempdb..test_class t1
INNER JOIN
tempdb..test_student t2
ON
t1.class_id = t2.class_id
WHERE
t1.class_id = #{classId}
</select>
<resultMap id="studentsFromOneClassMap" type="mybatistest.Student">
<id property="studentId" column="student_id" />
<result property="classId" column="class_id" />
<result property="studentName" column="student_name" />
<result property="studentAge" column="student_age" />
<result property="studentAddress" column="student_address" />
</resultMap> <!-- 获取一个年级的所有学生 -->
<select id="getStudentsFromOneGrade" resultMap="studentsFromOneGradeMap">
SELECT
t1.class_id,
t1.class_name,
t1.class_admin_name,
t2.student_id,
t2.student_name,
t2.student_age,
t2.student_address
FROM
tempdb..test_class t1
INNER JOIN
tempdb..test_student t2
ON
t1.class_id = t2.class_id
WHERE
t1.class_id/100 = #{gradeId}
</select>
<resultMap id="studentsFromOneGradeMap" type="mybatistest.StudentsPerClass">
<id property="classId" column="class_id"/>
<association property="clazz" javaType="mybatistest.Clazz" >
<id property="classId" column="class_id"/>
<result property="className" column="class_name"/>
<result property="classAdminName" column="class_admin_name"/>
</association>
<collection property="students" ofType="mybatistest.Student">
<id property="studentId" column="student_id" />
<result property="classId" column="class_id"/>
<result property="studentName" column="student_name" />
<result property="studentAge" column="student_age" />
<result property="studentAddress" column="student_address" />
</collection>
</resultMap> <!-- 获取所有学生 -->
<select id="getAllStudents" resultMap="allStudentsMap">
SELECT
t1.class_id,
t1.class_name,
t1.class_admin_name,
t2.student_id,
t2.student_name,
t2.student_age,
t2.student_address
FROM
tempdb..test_class t1
INNER JOIN
tempdb..test_student t2
ON
t1.class_id = t2.class_id
</select>
<resultMap id="allStudentsMap" type="mybatistest.StudentsPerClass">
<id property="classId" column="class_id"/>
<association property="clazz" javaType="mybatistest.Clazz" >
<id property="classId" column="class_id"/>
<result property="className" column="class_name"/>
<result property="classAdminName" column="class_admin_name"/>
</association>
<collection property="students" ofType="mybatistest.Student">
<id property="studentId" column="student_id" />
<result property="classId" column="class_id"/>
<result property="studentName" column="student_name" />
<result property="studentAge" column="student_age" />
<result property="studentAddress" column="student_address" />
</collection>
</resultMap>
</mapper>

mybatis-config.xml中添加映射关系

     <mappers>
<mapper resource="mybatistest/ClassStudentRepository.xml" />
</mappers>

<测试结果>

 package mybatistest;

 import java.util.List;
import java.util.Map; import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory; public class MainTest { public static void main(String[] args) {
SqlSessionFactory sqlSessionFactory = MybatisTestSessionFactory.getSqlSessionFactoryInstance();
SqlSession session = sqlSessionFactory.openSession();
ClassStudentRepository classStudentRepository = session.getMapper(ClassStudentRepository.class);
Map<Integer, Student> res1 = classStudentRepository.getStudentsFromOneClass(102);
res1.forEach((p, q) -> {
System.out.println("Student ID: " + p + "; Student info: " + q);
});
// [Output]
// Student ID: 102001; Student info: Student [classId=102, studentId=102001, studentName=Name 4, studentAge=22, studentAddress=Address 4]
// Student ID: 102002; Student info: Student [classId=102, studentId=102002, studentName=Name 5, studentAge=21, studentAddress=Address 5]
List<StudentsPerClass> res2 = classStudentRepository.getStudentsFromOneGrade(1);
res2.forEach((p) -> {
System.out.println(p);
});
// [Output]
// StudentsPerClass [classId=101, clazz=Clazz [classId=101, className=Class 1, Grade 1, classAdminName=Mr. Zhang], students=
// --Student [classId=101, studentId=101001, studentName=Name 1, studentAge=20, studentAddress=Address 1]
// --Student [classId=101, studentId=101002, studentName=Name 2, studentAge=21, studentAddress=Address 2]
// --Student [classId=101, studentId=101003, studentName=Name 3, studentAge=20, studentAddress=Address 3]
// ]
// StudentsPerClass [classId=102, clazz=Clazz [classId=102, className=Class 2, Grade 1, classAdminName=Mr. Li], students=
// --Student [classId=102, studentId=102001, studentName=Name 4, studentAge=22, studentAddress=Address 4]
// --Student [classId=102, studentId=102002, studentName=Name 5, studentAge=21, studentAddress=Address 5]
// ]
Map<Integer, StudentsPerClass> res3 = classStudentRepository.getAllStudents();
res3.forEach((p, q) -> {
System.out.println("Class ID: " + p + System.lineSeparator() + q);
});
// [Output]
// Class ID: 101
// StudentsPerClass [classId=101, clazz=Clazz [classId=101, className=Class 1, Grade 1, classAdminName=Mr. Zhang], students=
// --Student [classId=101, studentId=101001, studentName=Name 1, studentAge=20, studentAddress=Address 1]
// --Student [classId=101, studentId=101002, studentName=Name 2, studentAge=21, studentAddress=Address 2]
// --Student [classId=101, studentId=101003, studentName=Name 3, studentAge=20, studentAddress=Address 3]
// ]
// Class ID: 102
// StudentsPerClass [classId=102, clazz=Clazz [classId=102, className=Class 2, Grade 1, classAdminName=Mr. Li], students=
// --Student [classId=102, studentId=102001, studentName=Name 4, studentAge=22, studentAddress=Address 4]
// --Student [classId=102, studentId=102002, studentName=Name 5, studentAge=21, studentAddress=Address 5]
// ]
// Class ID: 201
// StudentsPerClass [classId=201, clazz=Clazz [classId=201, className=Class 1, Grade 2, classAdminName=Mr. Zhao], students=
// --Student [classId=201, studentId=201001, studentName=Name 6, studentAge=23, studentAddress=Address 6]
// --Student [classId=201, studentId=201002, studentName=Name 7, studentAge=22, studentAddress=Address 7]
// ]
// Class ID: 202
// StudentsPerClass [classId=202, clazz=Clazz [classId=202, className=Class 2, Grade 2, classAdminName=Mr. Liu], students=
// --Student [classId=202, studentId=202001, studentName=Name 8, studentAge=22, studentAddress=Address 8]
// ]
session.close();
}
}

Mybatis高级结果映射的更多相关文章

  1. Mybatis 高级结果映射 ResultMap Association Collection

    在阅读本文章时,先说几个mybatis中容易混淆的地方: 1. mybatis中的列不是数据库里的列而是查询里的列,可以是别名(如 select user_name as userName,这时col ...

  2. 转:mybatis 高级结果映射(http://blog.csdn.net/ilovejava_2010/article/details/8180521)

    高级结果映射 MyBatis的创建基于这样一个思想:数据库并不是您想怎样就怎样的.虽然我们希望所有的数据库遵守第三范式或BCNF(修正的第三范式),但它们不是.如果有一个数据库能够完美映射到所有应用程 ...

  3. MyBatis从入门到精通(十一):MyBatis高级结果映射之一对多映射

    最近在读刘增辉老师所著的<MyBatis从入门到精通>一书,很有收获,于是将自己学习的过程以博客形式输出,如有错误,欢迎指正,如帮助到你,不胜荣幸! 本篇博客主要讲解MyBatis中如何使 ...

  4. MyBatis从入门到精通(九):MyBatis高级结果映射之一对一映射

    最近在读刘增辉老师所著的<MyBatis从入门到精通>一书,很有收获,于是将自己学习的过程以博客形式输出,如有错误,欢迎指正,如帮助到你,不胜荣幸! 本篇博客主要讲解MyBatis中实现查 ...

  5. MyBatis高级查询

    -------------------------siwuxie095 MyBatis 高级查询 1.MyBatis 作为一个 ORM 框架,也对 SQL 的高级查询做了支持, MyBatis 高级查 ...

  6. 【Mybatis高级映射】一对一映射、一对多映射、多对多映射

    前言 当我们学习heribnate的时候,也就是SSH框架的网上商城的时候,我们就学习过它对应的高级映射,一对一映射,一对多映射,多对多映射.对于SSM的Mybatis来说,肯定也是差不多的.既然开了 ...

  7. 六 mybatis高级映射(一对一,一对多,多对多)

    1  订单商品数据模型 以订单商品数据为模型,来对mybaits高级关系映射进行学习.

  8. mybatis高级映射(一对一,一对多)

    mybatis高级映射 一对一关联映射 需求:查询订单信息,关联查询用户信息(一个订单对应一个用户) (1)通过resultType实现 sql语句: select orders.* , USER.u ...

  9. mybatis之高级结果映射

    先贴一句官方文档内容 如果世界总是这么简单就好了. 正如官方文档所说:如果一切都是这么简单,那该多好啊,但是实际上,我们面对的是复杂的对象,就是对象里有对象,有列表对象,总之五花八门的对象.这个时候我 ...

随机推荐

  1. java.util.NoSuchElementException

    问题引入 Java商店作业不同函数里需要获取用户输入,用Scanner的时候,出现了异常java.util.NoSuchElementException 作业中代码模式如下,func1和func2中都 ...

  2. MySQL 5.6, 5.7, 8.0版本的新特性汇总大全

    转载:http://blog.itpub.net/15498/viewspace-2650661/ MySQL 5.6 1).支持GTID复制 2).支持无损复制 3).支持延迟复制 4).支持基于库 ...

  3. SQL 归纳

    查询父节点的所有子节点: SELECT * FROM menu m START WITH m.ID_ = '402882836068695f0160688eebf70006' CONNECT BY m ...

  4. shell爬取斗图网

    #!/bin/bash read -p "请输入要爬取的页面数(默认为10):" page_num page_num=${page_num:-} echo $page_num re ...

  5. springboot 集成swagger2.x 后静态资源报404

    package com.bgs360.configuration; import org.springframework.context.EnvironmentAware; import org.sp ...

  6. springbootdruidmybatismysql多数据源事务管理

    springboot+druid+mybatis+mysql+多数据源事务管理 分布式事务在java中的解决方案就是JTA(即Java Transaction API):springboot官方提供了 ...

  7. 【luogu4781】拉格朗日插值

    题目背景 这是一道模板题 题目描述 由小学知识可知,nn个点(x_i,y_i)(xi​,yi​)可以唯一地确定一个多项式 现在,给定nn个点,请你确定这个多项式,并将kk代入求值 求出的值对99824 ...

  8. 阿里云Ubuntu安装LNMP环境之PHP7

    在QQ群很多朋友问阿里云服务器怎么安装LNMP环境,怎么把项目放到服务器上面去,在这里,我就从头开始教大家怎么在阿里云服务器安装LNMP环境. 在这之前,我们先要知道什么是LNMP. L: 表示的是L ...

  9. VUE项目引入jquery

    既然写项目,那么少不了用jq,那我们就引入进来吧 1.因为已经安装了vue脚手架,所以需要在webpack中全局引入jquery 打开package.json文件,在里面加入这行代码,jquery后面 ...

  10. Tomcat7修改根路径应用

    大家想必遇到过这样的问题,同台机器上跑上多个tomcat 那么随之更改文件的工作量就会增加 今天突然想到把所有的tomcat的根目录更改成一个文件 但是有不好的就的地方就是在更改文件的时候需要把这台机 ...