Mybatis关联查询之一对多和多对一XML配置详解

平时在开发过程中dao、bean和XML文件都是自动生成的,很少写XML的配置关系,今天记录一下mybatis的关联查询中的多对一和一对多的情况。

首先是有两张表(学生表Student和老师Teacher表,注:这里只是为了演示一对多和多对一的情况,请不要杠),为了更易懂,这里只设置了最简单的几个必要字段。表结构如下图

Student表:

Teacher表:

创建实体bean

Teacher.java:

import java.util.List;

public class Teacher {

 private Integer id;
 private String name;
 private String className;
 private List<Student> students;

 // get、set方法省略

}

Sfudent.java

public class Student {

 private Integer id;
 private String name;
 private Integer teacherId;
 private String className;
 private Teacher teacher;

 // get、set方法省略
}

下面重点来了:配置Mapper.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.tz.mybatis.dao.studentDao"> 

 <!-- ///一对多的第一种写法/// -->
 <resultMap type="Teacher" id="teacherMap">
  <id column="id" property="id"/>
  <result column="name" property="name"/>
   <collection property="students" ofType="Student" column="id">
    <!-- 这里的column对应的是下面查询的别名,而不是表字段名 -->
  <id column="sid" property="id"/>
    <!-- property对应JavaBean中的属性名 -->
  <result column="sname" property="name"/>
  <result column="className" property="className"/>
  </collection>
 </resultMap>

 <!-- 查询所有的老师级各自的所有学生 -->
 <select id="getTeachers" parameterType="Teacher" resultMap="teacherMap">
 SELECT
 t.id,
 t.NAME,
 t.class_Name,
 s.id AS sid,
 s. NAME AS sname,
 s.class_name as className
 FROM
 teacher t
 LEFT JOIN student s ON t.id = s.teacher_id
 </select>
</mapper>

测试类:

package com.tz.test;

import java.io.IOException;
import java.io.InputStream;
import java.util.List;
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.Before;
import org.junit.Test;
import com.tz.mybatis.bean.Student;
import com.tz.mybatis.bean.Teacher;

public class TeacherTest {

 private SqlSessionFactory sqlSessionFactory;

 @Before
 public void init() throws IOException {
 String resource = "mybatis-config.xml";
 InputStream inputStream = Resources.getResourceAsStream(resource);
 sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
 }

 @Test
 public void getTeachers() {
 SqlSession session = sqlSessionFactory.openSession();
 List<Teacher> list = session.selectList("com.tz.mybatis.dao.studentDao.getTeachers");
 System.out.println(list);
 }

}

下面给出第二种写法:

<!-- //一对多的第二种写法/ -->
 <resultMap type="Teacher" id="teacherMaps">
 <id column="id" property="id"/>
 <result column="name" property="name"/>
 <result column="class_name" property="className"/>
 <collection property="students" ofType="Student" select="getStudents" column="id">
 </collection>
 </resultMap>

 <!-- 查询所有的老师级各自的所有学生 -->
 <select id="getAllTeacher" parameterType="Teacher" resultMap="teacherMaps">
 SELECT
 t.id,
 t.NAME,
 t.class_name
 FROM
 teacher t
 </select>

 <select id="getStudents" parameterType="int" resultType="Student">
 select
 s.id,
 s. NAME,
 s.class_name as className
 from student s
 where teacher_id = #{id}
 </select>

测试类:

@Test
 public void getTeachers2() {
 SqlSession session = sqlSessionFactory.openSession();
 List<Teacher> list = session.selectList("com.tz.mybatis.dao.studentDao.getAllTeacher");
 System.out.println(list);
 }

查询学生信息(多对一):

首先还是配置文件:

<resultMap type="Student" id="studentMap">
 <id column="id" property="id"/>
 <result column="name" property="name"/>
 <result column="class_name" property="className"/>
 <result column="teacher_id" property="teacherId"/>
 <association property="teacher" select="getTeacher" column="teacher_id" javaType="Teacher">
 <!-- 这里要注意的是column对应的是student中的外键,而且需是表字段名 -->
 </association>
 </resultMap>

 <select id="getStudent" resultMap="studentMap">
 SELECT
 s.id,
 s.name,
 s.class_name,
 s.teacher_id
 FROM
 student s
 </select>

 <select id="getTeacher" resultType="Teacher" parameterType="int">
 SELECT
 t.id,
 t.name,
 t.class_name as className
 FROM teacher t
 where id = #{teacher_id}
 </select>

测试类:

@Test
 public void getStudents() {
 SqlSession session = sqlSessionFactory.openSession();
 List<Student> list = session.selectList("com.tz.mybatis.dao.studentDao.getStudent");
 System.out.println(list);
 }

最后:当然如果不想配置这么麻烦的信息,可以直接写一个关联查询的SQL语句,返回结果直接由Map接受即可。不过这样就不太符合面向对象的理念了。

到此这篇关于Mybatis关联查询之一对多和多对一XML配置详解的文章就介绍到这了,更多相关Mybatis关联查询内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

时间: 2020-10-18

Spring boot2基于Mybatis实现多表关联查询

模拟业务关系: 一个用户user有对应的一个公司company,每个用户有多个账户account. spring boot 2的环境搭建见上文:spring boot 2整合mybatis 一.mysql创表和模拟数据sql CREATE TABLE IF NOT EXISTS `user` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(50) NOT NULL, `company_id` int(11) NOT NULL, PRI

MyBatis 三表外关联查询的实现(用户、角色、权限)

一.数据库结构 二.查询所有数据记录(SQL语句) SQL语句: SELECT u.*, r.*, a.* FROM ( ( ( user u INNER JOIN user_role ur ON ur.user_id = u.user_id ) INNER JOIN role r ON r.role_id = ur.role_id ) INNER JOIN role_authority ra ON ra.role_id = r.role_id ) INNER JOIN authority a

MyBatis实践之动态SQL及关联查询

序言 MyBatis,大家都知道,半自动的ORM框架,原来叫ibatis,后来好像是10年apache软件基金组织把它托管给了goole code,就重新命名了MyBatis,功能相对以前更强大了.它相对全自动的持久层框架Hibernate,更加灵活,更轻量级,这点我还是深有体会的. MyBatis的一个强大特性之一就是动态SQL能力了,能省去我们很多串联判断拼接SQL的痛苦,根据项目而定,在一定的场合下使用,能大大减少程序的代码量和复杂程度,不过还是不是过度太过复杂的使用,以免不利于后期的维护

Mybatis多表关联查询的实现(DEMO)

概要 本节要实现的是多表关联查询的简单demo.场景是根据id查询某商品分类信息,并展示该分类下的商品列表. 一.Mysql测试数据 新建表Category(商品分类)和Product(商品),并插入几条测试数据. create table Category ( Id int not null auto_increment, Name varchar(80) null, constraint pk_category primary key (Id) ); INSERT INTO category

实例讲解Java的MyBatis框架对MySQL中数据的关联查询

mybatis 提供了高级的关联查询功能,可以很方便地将数据库获取的结果集映射到定义的Java Bean 中.下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的. 设计一个简单的博客系统,一个用户可以开多个博客,在博客中可以发表文章,允许发表评论,可以为文章加标签.博客系统主要有以下几张表构成: Author表:作者信息表,记录作者的信息,用户名和密码,邮箱等. Blog表   :  博客表,一个作者可以开多个博客,即Author和Blog的关系是一对多.

Mybatis 一对多和多对一关联查询问题

首先  数据库量表之间字段关系(没有主外键) studentmajor表的id字段对应student表里major字段 两个实体类 package com.model; import java.util.Date; public class Student { private Integer sno; private String sname; private String ssex; private Integer sclass; private StudentMajor studentmaj

Yii2中hasOne、hasMany及多对多关联查询的用法详解

前言 hasOne.hasMany是Yii2特有的用于多表关联查询的函数,平时在使用多表关联查询的时候建议使用它们.为什么?因为这种方式关联查询出来的结果会保留Yii2自有的表头排序功能,以及CheckboxColumn中input的id存值,至于还有没有其它的好处就需要大家去挖掘了,笔者目前就发现了这两个常用的好处.其他的关联查询,像yiidbQuery查询或者原生的SQL语句查询都没有,查询出来在列表展示的时候,表头一排黑. Yii2的hasOne.hasMany多表关联查询,不管是文档还是

Mybatis中的高级映射一对一、一对多、多对多

学习hibernate的时候,小编已经接触多各种映射,mybatis中映射有到底是如何运转的,今天这篇博文,小编主要来简单的介绍一下mybatis中的高级映射,包括一对一.一对多.多对多,希望多有需要的小伙伴有帮助,小编主要从四个方面进行介绍,订单商品数据模型.一对一查询.一对多查询.多对多查询. 一.订单商品数据模型 1.数据库执行脚本,如下所示: <span style="font-family:Comic Sans MS;font-size:18px;">CREATE

mybatis 一对一、一对多和多对多查询实例代码

关键字:association 一对一映射(一个班级只有一个班主任) <select id="getClass" parameterType="int" resultMap="ClassesResultMap"> select * from class c,teacher t where c.teacher_id=t.t_id and c.c_id=#{id} </select> <resultMap type=&q

mybatis多对多关联实战教程(推荐)

MyBatis3.0 添加了association和collection标签专门用于对多个相关实体类数据进行级联查询,但仍不支持多个相关实体类数据的级联保存和级联删除操作 一.创建student.teacher和stu_teach_rel三张张表 DROP TABLE IF EXISTS `student`; CREATE TABLE `student` ( `id` int(11) NOT NULL, `name` varchar(255) DEFAULT NULL, `age` int(11

mybatis一对多查询功能

首先,我们还是先给出一个需求:根据订单id查询订单明细--我们知道,一个订单里面可以有多个订单的明细(需求不明确的同学,请留言或者去淘宝网上的订单处点一下就知道了).这个时候,一个订单,对应多个订单的id.这种需求出现的时候,我们应该如何查询呢? 此时我们的数据模型如下图(左)由于查询用户也是我们的需求,所以就在原有的基础上进行扩展,数据模型如下(右): 很显然,如果用resultType的方式去实现的话,是不合理的了.因为我们需要创建一个既有订单又有订单明细的pojo然后呢,我们的mybati