MyBatis一对多关联查询

Java-框架王 MyBatis 发布时间:2021-07-20 16:06:28 阅读数:2311 1

一对多级联关系的处理

一对多的级联关系:在我们日常开发中有很多这样的情况,
如:
一个用户下面会有很多订单,但是每一个订单都只会属于一个用户,
这就是一个典型的一对多的级联关系操作,那么如何使用MyBatis进行此类数据的查询操作呢?下文将一一道来,如下所示:
实现思路:
  在 MyBatis 中,

我们可通过 <resultMap> 元素的子元素 <collection> 处理一对多级联关系,
collection 可以将关联查询的多条记录映射到一个 list 集合属性中,
代码如下
<collection property="orderList"
        ofType="com.java265.po.Order" column="id"
        select="com.java265.mapper.OrderMapper.selectOrderById" />

在<collection>元素中通常使用以下属性。 property:指定映射到实体类的对象属性。 column:指定表中对应的字段(即查询返回的列名)。 javaType:指定映射到实体对象属性的类型。 select:指定引入嵌套查询的子 SQL 语句,该属性用于关联映射中的嵌套查询。
  1. 一对多关联查询可采用以下两种方式:
  2. 分步查询,通过两次或多次查询,为一对多关系的实体 Bean 赋值
  3. 单步查询,通过关联查询实现
例 下面以用户和订单为例讲解一对多关联查询(实现“根据 id 查询用户及其关联的订单信息”的功能)的处理过程。

创建数据表

本实例需要两张数据表,一张是用户表 user,一张是订单表 order,这两张表具有一对多的级联关系。SQL 语句如下: CREATE TABLE `order` ( `id` int(11) NOT NULL AUTO_INCREMENT, `ordernum` int(25) DEFAULT NULL, `userId` int(11) DEFAULT NULL, PRIMARY KEY (`id`), KEY `userId` (`userId`) ) ENGINE=InnoDB AUTO_INCREMENT=9 DEFAULT CHARSET=utf8; insert into `order`(`id`,`ordernum`,`userId`) values (1,20210101,1),(2,20210102,2),(3,20210103,3),(4,20200645,1),(5,20210104,2),(6,20210105,2),(7,20210106,3); DROP TABLE IF EXISTS `user`; CREATE TABLE `user` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(20) DEFAULT NULL, `pwd` varchar(20) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8; insert into `user`(`id`,`name`,`pwd`) values (1,'张三','123'),(2,'王二','456'),(3,'麻子','123'),(4,'张积粮','345'),(5,'李晓爱','123'),(6,'古天乐','789');

创建持久化类

创建持久化类 User 和 Order,代码
package com.java265.po;
import java.util.List;
public class User {
    private int id;
    private String name;
    private String pwd;
    private List<Order> orderList;
    /*省略setter和getter方法*/
    @Override
    public String toString() {
        return "User [id=" + id + ", name=" + name + ", orderList=" + orderList + "]";
    }
}
Order 类
package com.java265.po;
public class Order {
    private int id;
    private int ordernum;
    /*省略setter和getter方法*/
    @Override
    public String toString() {
        return "Order [id=" + id + ", ordernum=" + ordernum + "]";
    }
}

分步查询

OrderMapper 类

public List<Order> selectOrderById(int id);
OrderMapper.xml 中相应的映射 SQL 语句

<select id="selectOrderById" resultType="com.java265.po.Order"
    parameterType="Integer">
    SELECT * FROM `order` where userId=#{id}
</select>

UserMapper 类
public User selectUserOrderById1(int id);
UserMapper.xml 中相应的映射 SQL 语句如下
<!-- 一对多 根据id查询用户及其关联的订单信息:级联查询的第一种方法(分步查询) -->
<resultmap id="userAndOrder1" type="com.java265.po.User">
    <id column="id" property="id">
    <result column="name" property="name">
    <result column="pwd" property="pwd">
    <!-- 一对多级联查询,ofType表示集合中的元素类型,将id传递给selectOrderById -->
    <collection column="id" oftype="com.java265.po.Order" property="orderList" select="com.java265.mapper.OrderMapper.selectOrderById">
</collection></result></result></id></resultmap>
<select id="selectUserOrderById1" parametertype="Integer" resultmap="userAndOrder1">
    select * from user where id=#{id}
</select>

测试代码
public class Test {
    public static void main(String[] args) throws IOException {
        InputStream config = Resources.getResourceAsStream("mybatis-config.xml");
        SqlSessionFactory ssf = new SqlSessionFactoryBuilder().build(config);
        SqlSession ss = ssf.openSession();
        User us = ss.getMapper(UserMapper.class).selectUserOrderById1(1);
        System.out.println(us);
    }
}
运行结果
DEBUG [main] - ==> Preparing: select * from user where id=?
DEBUG [main] - ==> Parameters: 1(Integer)
DEBUG [main] - ====> Preparing: SELECT * FROM `order` where userId=?
DEBUG [main] - ====> Parameters: 1(Integer)
DEBUG [main] - <==== Total: 2
DEBUG [main] - <== Total: 1 User [id=1, name=张三, orderList=[Order [id=1, ordernum=20210101], Order [id=4, ordernum=20200645]]]

单步查询

该种方式实现一对多关联查询需要修改 Order 持久化类,因为 Order 中的 id 不能和 User 中的 id 重复。
package com.java265.po;
public class Order {
    private int oId;
    private int ordernum;
    /*省略setter和getter方法*/
    @Override
    public String toString() {
        return "Order [id=" + oId+ ", ordernum=" + ordernum + "]";
    }
}
UserMapper 类
public User selectUserOrderById2(int id);
UserMapper.xml 中相关映射 SQL 语句

<!-- 一对多 根据id查询用户及其关联的订单信息:级联查询的第二种方法(单步查询) -->
<resultmap id="userAndOrder2" type="com.java265.po.User">
    <id column="id" property="id">
    <result column="name" property="name">
    <result column="pwd" property="pwd">
    <!-- 一对多级联查询,ofType表示集合中的元素类型 -->
    <collection oftype="com.java265.po.Order" property="orderList">
        <id column="oId" property="oId">
        <result column="ordernum" property="ordernum">
    </result></id></collection>
</result></result></id></resultmap>
<select id="selectUserOrderById2" parametertype="Integer" resultmap="userAndOrder2">
    SELECT u.*,o.id as oId,o.ordernum FROM `user` u,`order` o
    WHERE
    u.id=o.`userId` AND u.id=#{id}
</select>
测试代码修改调用方法,如下
public class Test {
    public static void main(String[] args) throws IOException {
        InputStream config = Resources.getResourceAsStream("mybatis-config.xml");
        SqlSessionFactory ssf = new SqlSessionFactoryBuilder().build(config);
        SqlSession ss = ssf.openSession();
        User us = ss.getMapper(UserMapper.class).selectUserOrderById2(1);
        System.out.println(us);
    }
}
运行结果如下
DEBUG [main] - ==> Preparing: SELECT u.*,o.id as oId,o.ordernum FROM `user` u,`order` o WHERE u.id=o.`userId` AND u.id=?
DEBUG [main] - ==> Parameters: 1(Integer)
DEBUG [main] - <== Total: 2 User [id=1, name=张三, orderList=[Order [id=1, ordernum=20200107], Order [id=4, ordernum=20200645]]]
版权声明

本文仅代表作者观点,不代表本站立场。
本文系作者授权发表,未经许可,不得转载。

本文链接: https://www.Java265.com/JavaFramework/MyBatis/202107/544.html

最近发表

热门文章

好文推荐

Java265.com

https://www.java265.com

站长统计|粤ICP备14097017号-3

Powered By Java265.com信息维护小组

使用手机扫描二维码

关注我们看更多资讯

java爱好者