MyBatis映射文件 <resultMap> 元素详解与示例


引言

<resultMap> 是 MyBatis 中最核心的映射配置元素,用于解决数据库字段与 Java 对象属性之间的复杂映射问题,尤其是字段名不一致、嵌套对象关联、集合映射等场景。ResultMap 的设计思想是,对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。以下是 <resultMap> 的详细解析和用法说明。


一、<resultMap> 基础结构

1. 核心属性
  • id:唯一标识符,用于其他操作引用此映射。
  • type:映射的目标 Java 对象类型(全限定类名或别名)。
  • autoMapping:是否启用自动映射(默认 false,建议部分场景开启)。
    在这里插入图片描述
2. 基础示例

场景:数据库字段为 user_id,Java 对象属性为 userId

<!-- 
用户结果映射(基础版本)
功能:将数据库查询结果映射到com.example.User对象
说明:
    1. id="BaseUserMap":标识这个结果映射的唯一名称。
    2. type="com.example.User":指定目标Java对象的完整类路径。
    3. <id property="userId" column="user_id">:定义主键属性userId与数据库列user_id的映射关系。
    4. <result property="userName" column="user_name">:定义普通属性userName与数据库列user_name的映射关系。
    5. <result property="age" column="age">:定义普通属性age与数据库列age的映射关系。
用法:在需要复杂映射的场景下,可复用此结果映射。
 -->
<resultMap id="BaseUserMap" type="com.example.User">
  <id property="userId" column="user_id" />
  <result property="userName" column="user_name" />
  <result property="age" column="age" />
</resultMap>

<!-- 
查询用户基本信息
功能:从users表中查询user_id、user_name和age列,并使用BaseUserMap结果映射转换为Java对象
说明:
    1. id="getUser":标识这个查询操作的唯一名称。
    2. resultMap="BaseUserMap":指定使用BaseUserMap结果映射来处理查询结果。
    3. SELECT user_id, user_name, age FROM users:从users表中查询指定列。
用法:在Mapper接口中定义对应的方法,例如:
    User getUser();
    返回一个包含用户基本信息的User对象。
 -->
<select id="getUser" resultMap="BaseUserMap">
  SELECT user_id, user_name, age FROM users
</select>

二、字段映射详解

1. <id><result> 的区别
  • <id>:标记主键字段,影响缓存和性能优化(必须配置)。
  • <result>:普通字段映射。
    在这里插入图片描述
2. 显式映射 vs 自动映射
  • 显式映射:手动配置所有字段(推荐复杂场景)。
  • 自动映射:开启 autoMapping="true" 后,未配置的字段按名称自动映射。
<!-- 
自动映射主键的用户结果映射
功能:使用MyBatis的自动映射功能,将数据库查询结果映射到User对象
特别说明:
    1. autoMapping="true":表示开启MyBatis的自动映射功能。
        - MyBatis会自动将查询结果中与User对象属性名匹配的列映射到对应的属性上。
    2. <id property="userId" column="user_id">:手动配置主键映射,确保主键列与User对象的userId属性正确对应。
        - 通常主键列需要显式定义,其他非主键列可以依赖自动映射完成。
 -->
<resultMap id="AutoUserMap" type="User" autoMapping="true">
  <id property="userId" column="user_id" /> <!-- 只需配置主键 -->
</resultMap>

<!-- 
查询用户信息(使用自动映射)
功能:从users表中查询所有用户信息,并通过AutoUserMap结果映射转换为User对象。
特别说明:
    - 该查询语句只需简单地从表中选择所有列(SELECT *),因为开启了自动映射,
      MyBatis会自动处理列名与User对象属性的映射。
    - 如果某些列名与User对象属性名不一致,可以手动在resultMap中配置对应的映射关系。
 -->
<select id="getUser" parameterType="String" resultMap="AutoUserMap">
  SELECT * FROM users
</select>

三、关联关系映射

1. 一对一关联 <association>

场景:用户(User)与身份证(IDCard)一对一关联。
Java 对象

public class User {
  private Integer userId;
  private String userName;
  private IDCard idCard; // 一对一对象
}

映射配置

<!-- 
用户与身份证信息的结果映射
功能:将用户信息和其关联的身份证信息映射到User对象
说明:
    1. id="UserWithIDCardMap":标识这个结果映射的唯一名称。
    2. type="User":指定目标Java对象类型为User。
    3. <id property="userId" column="user_id">:定义User对象的userId属性与数据库表的user_id列对应,作为主键。
    4. <result property="userName" column="user_name">:定义User对象的userName属性与数据库表的user_name列对应。
    5. <association property="idCard" javaType="IDCard">
        - property="idCard":User对象中的idCard属性,表示用户的身份证信息。
        - javaType="IDCard":指定关联对象的Java类型为IDCard。
        - <id property="cardId" column="card_id">:定义IDCard对象的cardId属性与数据库表的card_id列对应,作为主键。
        - <result property="cardNumber" column="card_number">:定义IDCard对象的cardNumber属性与数据库表的card_number列对应。
        - 通过这个关联映射,可以将用户的身份证信息嵌套到User对象中。
 -->
<resultMap id="UserWithIDCardMap" type="User">
  <id property="userId" column="user_id"/>
  <result property="userName" column="user_name"/>
  <!-- 嵌套对象映射 -->
  <association property="idCard" javaType="IDCard">
    <id property="cardId" column="card_id"/>
    <result property="cardNumber" column="card_number"/>
  </association>
</resultMap>

<!-- 
查询用户及其身份证信息
功能:从users和id_card表中查询用户及身份证数据,并通过UserWithIDCardMap结果映射转换为Java对象
说明:
    1. id="getUserWithIDCard":标识这个查询操作的唯一名称。
    2. resultMap="UserWithIDCardMap":指定使用UserWithIDCardMap结果映射来处理查询结果。
    3. SELECT u.user_id, u.user_name, c.card_id, c.card_number FROM users u LEFT JOIN id_card c ON u.user_id = c.user_id:
        - 查询users表和id_card表的联合数据。
        - LEFT JOIN将users表和id_card表连接,连接条件为用户ID(u.user_id = c.user_id)。
        - 查询结果将包含用户的基本信息和其关联的身份证信息。
用法:在Mapper接口中定义对应的方法,例如:
    User getUserWithIDCard();
    返回一个包含用户及其身份证信息的User对象。
 -->
<select id="getUserWithIDCard" resultMap="UserWithIDCardMap">
  SELECT 
    u.user_id, u.user_name, 
    c.card_id, c.card_number
  FROM users u
  LEFT JOIN id_card c ON u.user_id = c.user_id
</select>

2. 一对多关联 <collection>

场景:用户(User)与订单(Order)一对多关联。
Java 对象

public class User {
  private Integer userId;
  private String userName;
  private List<Order> orders; // 一对多集合
}

映射配置

<!-- 
用户与订单的结果映射
功能:将用户信息及其关联的订单信息映射到User对象
说明:
    1. id="UserWithOrdersMap":标识这个结果映射的唯一名称。
    2. type="User":指定目标Java对象类型为User。
    3. <id property="userId" column="user_id">:定义User对象的userId属性与数据库表的user_id列对应,作为主键。
    4. <result property="userName" column="user_name">:定义User对象的userName属性与数据库表的user_name列对应。
    5. <collection property="orders" ofType="Order">
        - property="orders":User对象中的orders属性,表示用户的订单集合。
        - ofType="Order":指定集合中元素的类型为Order。
        - <id property="orderId" column="order_id">:定义Order对象的orderId属性与数据库表的order_id列对应,作为主键。
        - <result property="orderNo" column="order_no">:定义Order对象的orderNo属性与数据库表的order_no列对应。
        - <result property="amount" column="amount">:定义Order对象的amount属性与数据库表的amount列对应。
        - 通过这个集合映射,可以将用户的订单信息作为嵌套集合封装到User对象中。
 -->
<resultMap id="UserWithOrdersMap" type="User">
  <id property="userId" column="user_id"/>
  <result property="userName" column="user_name"/>
  <!-- 集合映射 -->
  <collection property="orders" ofType="Order">
    <id property="orderId" column="order_id"/>
    <result property="orderNo" column="order_no"/>
    <result property="amount" column="amount"/>
  </collection>
</resultMap>

<!-- 
查询用户及其订单信息
功能:从users和orders表中查询用户及订单数据,并通过UserWithOrdersMap结果映射转换为Java对象
说明:
    1. id="getUserWithOrders":标识这个查询操作的唯一名称。
    2. resultMap="UserWithOrdersMap":指定使用UserWithOrdersMap结果映射来处理查询结果。
    3. SELECT u.user_id, u.user_name, o.order_id, o.order_no, o.amount FROM users u LEFT JOIN orders o ON u.user_id = o.user_id:
        - 查询users表和orders表的联合数据。
        - LEFT JOIN将users表和orders表连接,连接条件为用户ID(u.user_id = o.user_id)。
        - 查询结果将包含用户的基本信息和其关联的订单信息。
用法:在Mapper接口中定义对应的方法,例如:
    User getUserWithOrders();
    返回一个包含用户及其订单信息的User对象。
 -->
<select id="getUserWithOrders" resultMap="UserWithOrdersMap">
  SELECT 
    u.user_id, u.user_name,
    o.order_id, o.order_no, o.amount
  FROM users u
  LEFT JOIN orders o ON u.user_id = o.user_id
</select>

四、高级用法

1. 继承映射 (extends)

复用已有的 resultMap,避免重复配置。

<!-- 
基础用户结果映射
功能:定义了一个基础的结果映射,包含用户的基本字段(user_id 和 user_name)
说明:
    1. id="BaseUserMap":标识这个结果映射的唯一名称。
    2. type="User":指定目标Java对象类型为User。
    3. <id property="userId" column="user_id">:定义User对象的userId属性与数据库表的user_id列对应,作为主键。
    4. <result property="userName" column="user_name">:定义User对象的userName属性与数据库表的user_name列对应。
    5. 这个基础映射可以被其他映射继承和扩展,避免重复定义公共字段。
 -->
<resultMap id="BaseUserMap" type="User">
  <id property="userId" column="user_id"/>
  <result property="userName" column="user_name"/>
</resultMap>

<!-- 
扩展基础用户结果映射,添加新字段
功能:在基础用户映射的基础上,添加一个新的字段(email)
说明:
    1. id="ExtendedUserMap":标识这个结果映射的唯一名称。
    2. extends="BaseUserMap":表示继承BaseUserMap中的所有字段和配置。
    3. type="User":指定目标Java对象类型为User。
    4. <result property="email" column="email">:新增字段,定义User对象的email属性与数据库表的email列对应。
    5. 通过继承基础映射,可以避免重复定义公共字段,简化配置。
 -->
<resultMap id="ExtendedUserMap" extends="BaseUserMap" type="User">
  <result property="email" column="email"/>
</resultMap>

2. 构造函数映射 <constructor>

通过构造方法注入字段(适合不可变对象)。
Java 对象

public class User {
  private final Integer userId;
  private final String userName;
  
  public User(Integer userId, String userName) {
    this.userId = userId;
    this.userName = userName;
  }
}

映射配置

<!-- 
用户结果映射(基于构造函数)
功能:将数据库查询结果通过构造函数映射到User对象
说明:
    1. id="UserConstructorMap":标识这个结果映射的唯一名称。
    2. type="User":指定目标Java对象类型为User。
    3. <constructor>:表示将通过User类的构造函数来初始化对象。
    4. <arg column="user_id" javaType="int">:
        - 定义构造函数的第一个参数,对应数据库列user_id,Java类型为int。
    5. <arg column="user_name" javaType="String">:
        - 定义构造函数的第二个参数,对应数据库列user_name,Java类型为String。
    6. 这个结果映射要求User类有一个接受两个参数(int user_id, String user_name)的构造函数。
用法:在需要使用构造函数映射的场景下,通过这个resultMap将查询结果转换为User对象。
 -->
<resultMap id="UserConstructorMap" type="User">
  <constructor>
    <arg column="user_id" javaType="int"/>
    <arg column="user_name" javaType="String"/>
  </constructor>
</resultMap>

3. 嵌套查询(分步查询)

解决关联查询的 N+1 性能问题,通过分步加载数据。

<!-- 
用户与订单的延迟加载映射 先查用户,再通过 user_id 查订单
功能:定义了一个结果映射,用于将用户信息和其关联的订单信息进行映射。
      在查询用户时,不会立即加载订单信息,而是通过延迟加载的方式在需要时再加载。
说明:
    1. id="UserWithLazyOrdersMap":标识这个结果映射的唯一名称。
    2. type="User":指定目标Java对象类型为User。
    3. <id property="userId" column="user_id">:定义User对象的userId属性与数据库表的user_id列对应,作为主键。
    4. <result property="userName" column="user_name">:定义User对象的userName属性与数据库表的user_name列对应。
    5. <collection property="orders" ofType="Order" select="com.example.OrderMapper.getOrdersByUserId" column="user_id">
        - property="orders":User对象中的orders属性,表示用户关联的订单集合。
        - ofType="Order":指定集合中元素的类型为Order。
        - select="com.example.OrderMapper.getOrdersByUserId":指定一个单独的SQL查询语句,用于根据用户ID查询订单信息。
        - column="user_id":将查询结果中的user_id列作为参数传递给子查询。
 -->
<resultMap id="UserWithLazyOrdersMap" type="User">
  <id property="userId" column="user_id"/>
  <result property="userName" column="user_name"/>
  <collection 
    property="orders" 
    ofType="Order"
    select="com.example.OrderMapper.getOrdersByUserId"
    column="user_id"/> <!-- 将 user_id 作为参数传递给子查询 -->
</resultMap>

<!-- 
查询用户信息(延迟加载订单)
功能:根据用户ID查询用户的基本信息,并通过延迟加载的方式获取其关联的订单信息。
说明:
    1. id="getUser":标识这个查询操作的唯一名称。
    2. resultMap="UserWithLazyOrdersMap":指定使用UserWithLazyOrdersMap结果映射来处理查询结果。
    3. SELECT user_id, user_name FROM users WHERE user_id = #{id}:
        - 查询users表中的user_id和user_name列。
        - WHERE user_id = #{id}:根据传入的用户ID进行查询。
用法:在Mapper接口中定义对应的方法,例如:
    User getUser(int id);
    返回一个包含用户基本信息的User对象,其orders属性在需要时才会加载。
 -->
<select id="getUser" resultMap="UserWithLazyOrdersMap">
  SELECT user_id, user_name FROM users WHERE user_id = #{id}
</select>

OrderMapper.xml

<!-- 
根据用户ID查询订单信息
功能:从orders表中查询指定用户的所有订单
参数:userId(用户ID)
返回:Order对象的列表,包含该用户的所有订单信息
说明:
    1. id="getOrdersByUserId":标识这个查询操作的唯一名称。
    2. resultType="Order":指定查询结果的类型为Order对象。
    3. SELECT * FROM orders WHERE user_id = #{userId}:
        - 查询orders表中所有列的数据。
        - WHERE user_id = #{userId}:条件是订单的user_id等于传入的userId参数。
用法:在Mapper接口中定义对应的方法,例如:
    List<Order> getOrdersByUserId(int userId);
    调用时传入用户ID,返回该用户的所有订单信息。
 -->
<select id="getOrdersByUserId" resultType="Order">
  SELECT * FROM orders WHERE user_id = #{userId}
</select>

在这里插入图片描述


五、常见问题与最佳实践

  1. 主键必须配置<id> 影响缓存机制,未配置可能导致性能下降。
  2. 避免过度嵌套:复杂关联建议使用分步查询(嵌套查询)减少单次 SQL 复杂度。
  3. 自动映射的取舍:简单场景可开启 autoMapping,复杂字段仍需显式配置。
  4. 性能优化:一对多关联使用懒加载(fetchType="lazy"),按需加载数据。

六、总结

拓展阅读:MyBatis映射文件常用元素详解与示例
官方文档:MyBatis | XML 映射文件
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/969099.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在ArcGIS JS API中使用WebGL实现波纹扩散特效

在现代WebGIS开发中&#xff0c;ArcGIS JS API 是一个非常强大的工具&#xff0c;它允许开发者创建丰富的地理信息应用。结合WebGL技术&#xff0c;我们可以实现更加复杂和炫酷的可视化效果。本文将介绍如何使用ArcGIS JS API结合WebGL实现一个波纹扩散特效。 波纹扩散效果 1 概…

基于Java的图书管理网站:SpringBoot+Vue开发的图书借阅管理系统

基于Java的图书管理网站&#xff1a;SpringBootVue开发的图书借阅管理系统 引言 随着信息技术的飞速发展&#xff0c;传统的图书借阅方式已逐渐向智能化、信息化方向转变。为了提高图书管理的效率和用户的借阅体验&#xff0c;基于Java、SpringBoot和Vue开发了一套图书借阅管…

MongoDB 7 分片副本集升级方案详解(上)

#作者&#xff1a;任少近 文章目录 前言&#xff1a;Mongodb版本升级升级步骤环境1.1环境准备1.2standalone升级1.3分片、副本集升级 前言&#xff1a;Mongodb版本升级 在开始升级之前&#xff0c;请参阅 MongoDB下个版本中的兼容性变更文档&#xff0c;以确保您的应用程序和…

Lua闭包的使用以及需要注意的问题

1. 闭包的基本概念 在 Lua 中&#xff0c;闭包是一个函数值&#xff0c;它包含了函数本身以及该函数所创建时的环境。闭包允许函数访问其外部函数作用域中的变量&#xff0c;即使外部函数已经执行完毕。 2.闭包的简单使用 代码&#xff1a;在下面的代码中&#xff0c;create…

第12周:LSTM(火灾温度)

1.库以及数据的导入 1.1库的导入 import torch.nn.functional as F import numpy as np import pandas as pd import torch from torch import nn1.2数据集的导入 data pd.read_csv("woodpine2.csv")dataTimeTem1CO 1Soot 100.00025.00.0000000.00000010.22825.…

日志结构化处理:PO对象toString日志转JSON工具

日志结构化处理&#xff1a;PO对象toString日志转JSON工具 1. 解决的问题2. 下载地址 在Java项目中&#xff0c;PO&#xff08;Plain Old Java Object&#xff09;对象遍布各个角落&#xff0c;且常常伴随着大量的日志记录需求。传统的做法是通过toString方法直接打印这些对象&…

QML 快捷键与Shortcut的使用

一、效果展示 二、源码分享 import QtQuick import QtQuick.Controls import Qt.labs.qmlmodels import QtQuick.Controls.Basic import QtQuick.Layouts import QtQuick.Effects import Qt.labs.platformApplicationWindow {id:rootwidth: 1000height: 730visible: truetitle…

蓝桥杯之并查集

算法思想 并查集是一种树形的数据结构&#xff0c;主要用于解决一些元素分组问题。用于处理一些不相交集合的合并以及查询问题。并查集的思想是用一个数组表示了整片森林&#xff0c;树的根节点唯一标识了一个集合&#xff0c;我们只要找到了某个元素的树根&#xff0c;就能确…

Qt多线程技术【线程池】:QRunnable 和 QThreadPool

在现代软件开发中&#xff0c;尤其是在处理大量并发任务时&#xff0c;线程池技术是一种高效的解决方案。线程池不仅能提高程序的性能&#xff0c;还能有效管理线程的生命周期&#xff0c;避免频繁的线程创建和销毁所带来的性能损失。本文将以Qt中的 QThreadPool 和 QRunnable …

链表 —— 常用技巧与操作总结详解

引言 链表作为一种动态数据结构&#xff0c;以其灵活的内存管理和高效的插入删除操作&#xff0c;在算法与工程实践中占据重要地位。然而&#xff0c;链表的指针操作复杂&#xff0c;容易引发内存泄漏和野指针问题。本文博主将从基础操作到高阶技巧&#xff0c;系统化解析链表的…

Renesas RH850 FDL库介绍

文章目录 FDL库(Data Flash Library)简介FDL库的核心功能FDL库的使用步骤关键注意事项示例应用场景总结FDL库(Data Flash Library)简介 FDL(Data Flash Library)是Renesas为RH850系列微控制器提供的数据闪存(Data Flash)操作库,用于简化数据闪存的擦除、写入、读取等…

Linux 配置 MySQL 定时自动备份到另一台服务器

Linux 配置 MySQL 定时自动备份到另一台服务器 前言1、配置服务器通信1.1&#xff1a;配置过程 2、编写自动备份sh脚本文件3&#xff1a;设置定时自动执行 前言 此方案可使一台服务器上的 MySQL 中的所有数据库每天 0 点自动转储为 .sql 文件&#xff0c;然后将文件同步到另一…

用php tp6对接钉钉审批流的 table 表格 明细控件 旧版sdk

核心代码 foreach ($flows[product_list] as $k>$gift) {$items_list[] [[name > 商品名称, value > $gift[product_name] ?? ],[name > 规格, value > $gift[product_name] ?? ],[name > 数量, value > $gift[quantity] ?? ],[name > 单位, v…

RV1126解码(1)

比如我们现在要拉一个流&#xff0c; 拉一个rtmp或者拉一个rtsp的流&#xff0c;让它显示到显示屏上面去&#xff0c;此时就要用到我们这个解码模块了&#xff0c;把它个解出来并且发到其他模块去。 主要功能是通过FFMPEG的API读取每一帧的音视频数据&#xff0c;并通过RV1126的…

sql:时间盲注和boolen盲注

关于时间盲注&#xff0c;boolen盲注的后面几个获取表、列、具体数据的函数补全 时间盲注方法 import time import requests# 获取数据库名 def inject_database(url):dataname for i in range(1, 20):low 32high 128mid (low high) // 2while low < high:payload &q…

DeepSeek+Excel 效率翻倍

2025年初&#xff0c;DeepSeek以惊人的效率突破技术壁垒&#xff0c;用极低的成本实现了与行业顶尖AI相媲美的性能&#xff0c;瞬间成为全球科技领域的热门话题。 那么AI工具的普及将如何改变我们的工作方式&#xff1f;Excel会被取代吗&#xff1f; 今天&#xff0c;珠珠带你…

WPS或word接入智能AI

DeepSeek接入WPS 配置WPS &#xff08;1&#xff09;下载 OfficeAl助手插件: 插件下载地址:https://www.office-ai.cn/。 安装插件后&#xff0c;打开WPS&#xff0c;菜单栏会新增"OfficeAl助手”选项卡。 如果没有出现&#xff0c; 左上找到文件菜单 -> 选项 ,在…

论文学习记录之《CLR-VMB》

目录 一、基本介绍 二、介绍 三、方法 3.1 FWI中的数据驱动方法 3.2 CLR-VMB理论 3.3 注意力块 四、网络结构 4.1 网络架构 4.2 损失函数 五、实验 5.1 数据准备 5.2 实验设置 5.3 训练和测试 5.4 定量分析 5.5 CLR方案的有效性 5.6 鲁棒性 5.7 泛化性 六、讨…

使用 EDOT 监测由 OpenAI 提供支持的 Python、Node.js 和 Java 应用程序

作者&#xff1a;来自 Elastic Adrian Cole Elastic 很自豪地在我们的 Python、Node.js 和 Java EDOT SDK 中引入了 OpenAI 支持。它们为使用 OpenAI 兼容服务的应用程序添加日志、指标和跟踪&#xff0c;而无需任何代码更改。 介绍 去年&#xff0c;我们宣布了 OpenTelemetry…

Golang的多团队协作编程模式与实践经验

Golang的多团队协作编程模式与实践经验 一、多团队协作编程模式概述 在软件开发领域&#xff0c;多团队协作编程是一种常见的工作模式。特别是对于大型项目来说&#xff0c;不同团队间需要协同合作&#xff0c;共同完成复杂的任务。Golang作为一种高效、并发性强的编程语言&…