有2个实体:用户、订单,一个用户可以拥有多个订单,同时这多个订单属于一个用户,即一对多。

user_tb:

order_tb:

在“多”的一方(order)添加“一”的一方(user)的主键(user_id)作为外键。


使用嵌套结果

(1)给2个实体都编写pojo类,需要在“一”的一方写个List来关联“多”的一方

package com.chy.pojo;

public class Order {
private Integer no;
private Integer userId;
private Integer goodsId;
private Integer goodsAmount; public Integer getNo() {
return no;
} public void setNo(Integer no) {
this.no = no;
} public Integer getUserId() {
return userId;
} public void setUserId(Integer userId) {
this.userId = userId;
} public Integer getGoodsId() {
return goodsId;
} public void setGoodsId(Integer goodsId) {
this.goodsId = goodsId;
} public Integer getGoodsAmount() {
return goodsAmount;
} public void setGoodsAmount(Integer goodsAmount) {
this.goodsAmount = goodsAmount;
} @Override
public String toString() {
return "Order{" +
"no=" + no +
", userId=" + userId +
", goodsId=" + goodsId +
", goodsAmount=" + goodsAmount +
'}';
}
}
package com.chy.pojo;

import java.util.List;

public class User {
private Integer id;
private String username;
private String password;
private String tel;
private String address;
private List<Order> orderList; public Integer getId() {
return id;
} public void setId(Integer id) {
this.id = id;
} public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPassword() {
return password;
} public void setPassword(String password) {
this.password = password;
} public String getTel() {
return tel;
} public void setTel(String tel) {
this.tel = tel;
} public String getAddress() {
return address;
} public void setAddress(String address) {
this.address = address;
} public List<Order> getOrderList() {
return orderList;
} public void setOrderList(List<Order> orderList) {
this.orderList = orderList;
} @Override
public String toString() {
return "User{" +
"id=" + id +
", username='" + username + '\'' +
", password='" + password + '\'' +
", tel='" + tel + '\'' +
", address='" + address + '\'' +
", orderList=" + orderList +
'}';
}
}

toString()只是为了方便测试、调试,后期可以去掉。

(2)给“一”的一方写Mapper接口、xml映射文件

package com.chy.mapper;

import com.chy.pojo.User;

public interface UserMapper {
public User queryUserById(Integer id);
}
<?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.chy.mapper.UserMapper">
<select id="queryUserById" parameterType="integer" resultMap="userResultWithOrder">
SELECT user_tb.*,order_tb.* FROM user_tb,order_tb WHERE user_tb.id=#{id} AND order_tb.user_id=user_tb.id
</select>
<resultMap id="userResultWithOrder" type="user">
<id property="id" column="id"/>
<result property="username" column="username"/>
<result property="password" column="password"/>
<result property="tel" column="tel"/>
<result property="address" column="address"/>
<collection property="orderList" ofType="order">
<id property="no" column="no"/>
       <result property="userId" column="user_id" />
<result property="goodsId" column="goods_id" />
<result property="goodsAmount" column="goods_amount" />
</collection>
</resultMap>
</mapper>

因为关联的是集合(List),所以用<collection>。

  • property指定表示关联对象的成员变量名
  • ofType指定返回的数据类型,注意不是list。
  • 子元素<id>、<result>配置关联对象的映射。

所谓关联查询,是查询结果中同时包含双方(2张表)的多个字段,

如果只查询一张表的字段,比如说只查询某个用户的订单,不查询此用户本身的信息,直接根据外键user_id查order_tb表,不需要查user_tb,不必使用关联查询,写个OrderMapper接口、OrderMapper.xml就ok。

(3)使用

package com.chy.utils;

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 java.io.IOException;
import java.io.InputStream; public class MyBatisUtils {
private static SqlSessionFactory sqlSessionFactory; static {
try {
InputStream inputStream = Resources.getResourceAsStream("mybatis-config.xml");
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
} public static SqlSession getSqlSession(){
return sqlSessionFactory.openSession();
}
}
package com.chy.test;

import com.chy.mapper.UserMapper;
import com.chy.pojo.User;
import com.chy.utils.MyBatisUtils;
import org.apache.ibatis.session.*; public class Test {
public static void main(String[] args) {
SqlSession sqlSession = MyBatisUtils.getSqlSession();
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.queryUserById(1);
System.out.println(user);
sqlSession.close();
}
}

使用嵌套查询

与嵌套结果大体相同,不同的只有第二步:

(二)给双方都编写Mapper接口、映射文件

package com.chy.mapper;

import com.chy.pojo.Order;

public interface OrderMapper {
//根据orderId查询订单信息
public Order queryOrderByUserId(Integer userId);
}
<?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.chy.mapper.OrderMapper">
<select id="queryOrderByUserId" parameterType="integer" resultMap="orderMap">
SELECT * FROM order_tb WHERE user_id=#{id}
</select>
<resultMap id="orderMap" type="order">
<id property="no" column="no"/>
<result property="userId" column="user_id"/>
<result property="goodsId" column="goods_id"/>
<result property="goodsAmount" column="goods_amount"/>
</resultMap>
</mapper>
package com.chy.mapper;

import com.chy.pojo.User;

public interface UserMapper {
public User queryUserById(Integer id);
}
<?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.chy.mapper.UserMapper">
<select id="queryUserById" parameterType="integer" resultMap="userResultWithOrder">
SELECT * FROM user_tb WHERE id=#{id}
</select>
<resultMap id="userResultWithOrder" type="user">
<id property="id" column="id"/>
<result property="username" column="username"/>
<result property="password" column="password"/>
<result property="tel" column="tel"/>
<result property="address" column="address"/>
<collection property="orderList" column="id" ofType="order" select="com.chy.mapper.OrderMapper.queryOrderByUserId"/>
</resultMap>
</mapper>

column指定向子查询传递的参数是当前表的哪一列。


<resultMap>中的映射问题

我在<resultMap>中使用了子元素<id>、<result>配置了所有的表字段——pojo类属性之间的映射,

其实只要表字段、pojo类属性的名称一致,就不必配置,比如

<resultMap id="userResultWithOrder" type="user">
<id property="id" column="id"/>
<result property="username" column="username"/>
<result property="password" column="password"/>
<result property="tel" column="tel"/>
<result property="address" column="address"/>
<collection property="orderList" column="id" ofType="order" select="com.chy.mapper.OrderMapper.queryOrderByUserId"/>
</resultMap>

红色部分完全可以省略,因为默认的映射就是pojo类属性、表字段名称一致。

如果部分不一致,只需配置不一致的部分,比如只有用户id的不一致:

<resultMap id="userResultWithOrder" type="user">
<id property="id" column="user_id"/>
<collection property="orderList" column="id" ofType="order" select="com.chy.mapper.OrderMapper.queryOrderByUserId"/>
</resultMap>

MyBatis 关联查询的实现:一对多的更多相关文章

  1. MyBatis关联查询,一对多关联查询

    实体关系图,一个国家对应多个城市 一对多关联查询可用三种方式实现: 单步查询,利用collection标签为级联属性赋值: 分步查询: 利用association标签进行分步查询: 利用collect ...

  2. mybatis 关联查询实现一对多

    场景:最近接到一个项目是查询管理人集合  同时每一个管理人还存在多个出资人   要查询一个管理人列表  每个管理人又包含了出资人列表 采用mybatis关联查询实现返回数据. 实现方式: 1 .在实体 ...

  3. MyBatis基础:MyBatis关联查询(4)

    1. MyBatis关联查询简介 MyBatis中级联分为3中:association.collection及discriminator. ◊ association:一对一关联 ◊ collecti ...

  4. MyBatis关联查询、多条件查询

    MyBatis关联查询.多条件查询 1.一对一查询 任务需求; 根据班级的信息查询出教师的相关信息 1.数据库表的设计 班级表: 教师表: 2.实体类的设计 班级表: public class Cla ...

  5. mybatis关联查询基础----高级映射

    本文链接地址:mybatis关联查询基础----高级映射(一对一,一对多,多对多) 前言: 今日在工作中遇到了一个一对多分页查询的问题,主表一条记录对应关联表四条记录,关联分页查询后每页只显示三条记录 ...

  6. Mybatis关联查询和数据库不一致问题分析与解决

    Mybatis关联查询和数据库不一致问题分析与解决 本文的前提是,确定sql语句没有问题,确定在数据库中使用sql和项目中结果不一致. 在使用SpringMVC+Mybatis做多表关联时候,发现也不 ...

  7. Mybatis关联查询之二

    Mybatis关联查询之多对多 多对多 一.entity实体类 public class Student { private Integer stuid; private String stuname ...

  8. MyBatis关联查询和懒加载错误

    MyBatis关联查询和懒加载错误 今天在写项目时遇到了个BUG.先说一下背景,前端请求更新生产订单状态,后端从前端接收到生产订单ID进行查询,然后就有问题了. 先看控制台报错: org.apache ...

  9. mybatis关联查询,一对一,一对多

    注:这篇文章的代码有部分删减,不能直接使用,不过关键代码都存在  应用场景: 想用mybatis做关联查询,并且把查询出的数据自动组装成对象可以使用关联查询. 1.一对一实现 例如:一部小说,属于一个 ...

随机推荐

  1. java文件相关(文件追加内容、文件内容清空、文件内容读取)

    https://blog.csdn.net/xnz0616/article/details/39137177 1.文件内容追加 // 在已有的文件后面追加信息 public static void a ...

  2. Lesson 5 Youth

    How does the writer like to treat young people? People are always talking about 'the problem of yout ...

  3. numpy高级函数:where与extract

    numpy高级函数:where与extract 1.numpy.where()函数,此函数返回数组中满足某个条件的元素的索引: import numpy as np x = np.array([[1, ...

  4. 使用vim编译.cpp文件

    一.编写代码 1.打开命令行终端,输入vim test.cpp,新建了一个文件叫做“test.cpp”:如果以前已经建立过这个文件,则是打开这个名字的文件. 2.按回车进入编辑界面,输入i进入编辑模式 ...

  5. js 对象补充

    js 对象 普通对象与对象简写 js 没有字典类型,只有对象类型,对象可以完全替代字典来使用 js 中对象的属性名,都采用字符串类型,所以就可以省略字符串的引号标识 对象中属性值为函数时,称之为方法, ...

  6. 蓝牙 BLE 协议学习: 001-BLE协议栈整体架构

    背景 在深入BLE协议帧之前,我们先看一下BLE协议栈整体架构. 转载自:<深入浅出低功耗蓝牙(BLE)协议栈> 架构 如上图所述,要实现一个BLE应用,首先需要一个支持BLE射频的芯片, ...

  7. Heap(堆)的基础知识入门

    堆 逻辑结构: 1   /        \ 1          3 /     \     /    \ 4    5   6      null 物理结构; 1.首先堆是一个完全二叉查找书(Co ...

  8. Linux系统-----包管理器的演变

    每个电脑设备都使用某种形式的软件来执行其预定任务.在软件开发的早期,对产品进行了严格的bug和其他缺陷测试.在过去的十多年里,软件通过互联网发布,目的是通过应用新版本的软件来修复任何错误.在某些情况下 ...

  9. html5,css3炫酷实例-元素

    自动完成输入框下拉列表 使用的插件:jquery-ui 使用数据源实现文本框的自动完成功能 <link href="https://cdn.bootcss.com/jqueryui/1 ...

  10. 017.CI4框架CodeIgniter数据库操作之:Updata更新修改一条数据

    01. 在Model中写入数据库操作的代码如下: <?php namespace App\Models\System; use CodeIgniter\Model; class User_mod ...