mybatis通过中间表实现一对多查询功能

需求:
通过一个学生的id查询出该学生所学的所有科目。

使用到的表格:
1.student:学生表

2.subject:科目表

3.stu_sub:学生-科目表(这里的成绩字段没用到,不用管)

实体类( get、set方法省略):
1.student

public class Student implements Serializable {
    private int id;
    private String stuNum;  //学号
    private String password;
    private String stuName;
    private String grade;   //年级
    private String department;  //系
    private String professional;    //专业
    private List<Subject> subList; //修读课程
}

2.subject

public class Subject implements Serializable {
    private Integer id;
    private String subjectName;
    }

首先记一次错误的实践:
最开始的想法很美好,思路示例如下:
首先通过stu_sub语句块在中间表查询出学生id对应的subId,在通过resultMap里面的collection标签将subId传到stu_sub_1语句块中,实现联级查询。

结果:
可以查询出该学生对应的所有科目,但是无法封装到student实体类的List<subject.>属性中,会报错期望的返回值是1,但结果是n。
原因:
mybatis若是没有用主键来查询对多关系,就会把查询出来的多个结果当成多个对象来封装。如上述例子,第一步通过stu_sub语句块查询出来的结果为:

这时候由于resultMap的type是student类型,所以mybatis会把上述的4个stuId当成四个对象来存放collection查询出来的科目。但我们方法的返回值是Student,所以会因为无法存放4个student而报错。

-------------------------------------------------------------分割线----------------------------------------------------------------
下面是成功实现的示例:
直接使用left join语句将三个表关联查询,再通过resultMap将我们所需要的学科注入到student对象中。xml代码实现如下:

<resultMap id="Map_stu_sub" type="student">
        <id column="id" property="id"></id>
        <result column="stuNum" property="stuNum"></result>
        <result column="stuName" property="stuName"></result>
        <result column="grade" property="grade"></result>
        <result column="department" property="department"></result>
        <result column="professional" property="professional"></result>
        <collection property="subList" ofType="subject">
            <!--将subject的id改成其他别名,不然会跟student的id字段冲突-->
            <id column="sub_id" property="id"></id>
            <result column="subjectName" property="subjectName"></result>
        </collection>
    </resultMap>

    <select id="stu_sub" resultMap="Map_stu_sub">
       SELECT a.*,c.id as sub_id,c.subjectName FROM student a LEFT OUTER JOIN stu_sub b ON a.id=b.stuId LEFT JOIN SUBJECT c ON b.subId=c.id WHERE a.id = #{id}
    </select>

然后直接调用stu_sub方法即可。

到此这篇关于mybatis通过中间表实现一对多查询的文章就介绍到这了,更多相关mybatis一对多查询内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Mybatis自关联查询一对多查询的实现示例

    注:代码已托管在GitHub上,地址是:https://github.com/Damaer/Mybatis-Learning ,项目是mybatis-13-oneself-one2many,需要自取,需要配置maven环境以及mysql环境(sql语句在resource下的test.sql中),觉得有用可以点个小星星. docsify文档地址在:https://damaer.github.io/Mybatis-Learning/#/ 所谓自关联查询,是指自己既然充当一方,又充当多方.比如新闻栏目

  • Mybatis一对多查询的两种姿势(值得收藏)

    前言 最近碰到了Mybatis一对多查询的场景,在这里总结对比下常见的两种实现方式. 本文以常见的订单表和订单详情表来举例说明: 数据库表准备 订单表 tbl_order 订单详情表 tlb_order_detail ps: 一个订单关联多个订单详情,通过order_no订单号关联: 实例演示 方法一:联合查询ResultMap映射 sql直接关联查询,然后结果集通过resultMap的collection映射 例如 查询订单列表,包括订单详情 Order.java 中新增字段orderDeta

  • mybatis一对多查询功能

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

  • mybatis高级映射一对多查询实现代码

    1.需求分析: 在开发中会遇到这样一个问题,查询订单信息,级联查询出用户信息和订单明细信息 2.sql语句实现 2.1确定主查询表:订单表 2.2确定关联查询表:用户表, 订单明细表 sql语句如下: select orders.*, t_user.address, t_user.name, t_user.brithday, orderdetail.id orderdetail_id, orderdetail.orderid, orderdetail.itemsid from orders, t

  • mybatis通过中间表实现一对多查询功能

    需求: 通过一个学生的id查询出该学生所学的所有科目. 使用到的表格: 1.student:学生表 2.subject:科目表 3.stu_sub:学生-科目表(这里的成绩字段没用到,不用管) 实体类( get.set方法省略): 1.student public class Student implements Serializable { private int id; private String stuNum; //学号 private String password; private

  • mybatis水平分表实现动态表名的项目实例

    目录 一.水平分表 二.项目实现 目录结构 三.扩展 一.水平分表 当业务需求的数据量过大时,一个表格存储数据会非常之多,故时长采用水平分表的方式来减少每张表的数据量即是提升查询数据库时的效率. 水平分表时,各表的结构完全一样,表名不同. 例如:这里我们建了10张user表,每张表的结构完全一致,表名由0~9. 表中包含有id和name属性且都采用varchar的存储类型. 为什么id没有采用int自增的形式? 大型项目极有可能采用分布式数据库,若采用自增的方式,会导致id重复.且id也不一定只

  • 解析Mybatis对sql表的一对多查询问题

    Mybatisd对sql表的一对多查询 select * from projectrecord pr left join projects po on po.pid=pr.pid left join emp e on e.empno = pr.empno where pr.pid=1 <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.or

  • MyBatis图文并茂讲解注解开发一对多查询

    目录 MyBatis的注解实现复杂映射开发 一对多查询 一对多查询的模型 一对多查询的语句 创建StudentMapper接口 使用注解配置Mapper 测试类 一对多配置总结 MyBatis的注解实现复杂映射开发 实现复杂关系映射之前我们可以在映射文件中通过配置来实现,使用注解开发后,我们可以使用@Results注解,@Result注解,@One注解,@Many注解组合完成复杂关系的配置 一对多查询 一对多查询的模型 一对多查询的需求:查询一个课程,与此同时查询出该该课程对应的学生信息 一对多

  • springboot整合mybatis-plus基于注解实现一对一(一对多)查询功能

    因为目前所用mybatis-plus版本为3.1.1,感觉是个半成品,所有在实体类上的注解只能支持单表,没有一对一和一对多关系映射,且该功能还在开发中,相信mybatis-plus开发团队在不久的将来应该会实现此功能. 由于本人开发习惯的原因,实在是太讨厌大量的xml充斥在整个项目中,尤其是表的mapper.xml,虽然有代码生成器可以生成,但是有些复杂的查询还是需要手写配置文件里的动态sql,这点比较反感(至于为什么反感,也是有多方面原因的). 不过可能是大量的java开发人员已经被虐惯了,已

  • springboot整合mybatis实现简单的一对多级联查询功能

    本文的目的是用springboot整合mybatis实现一个简单的一对多查询.(查询一个用户有多少件衣服) 第一步:数据库中,可以直接在navicat中建立两张我们需要用到的表 users DROP TABLE IF EXISTS `users`; CREATE TABLE `users` ( `id` int(0) NOT NULL AUTO_INCREMENT, `name` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_

  • C语言使用顺序表实现电话本功能

     简介: 用顺序表实现电话本的功能(C语言) 电话本具有如下4个功能: 1.创建一个电话本,电话本里面包含名字和电话号码 2.在指定位置插入一个名字和电话号码 3.在指定位置删除一个名字和电话号码 4.打印电话本 代码: //其中那个color函数是我为了美观加上去的,如果感觉不需要的话可以将代码中所有有关color的都删掉即可 #include <iostream> #include <cstdio> #include <cstring> #include <a

随机推荐