Mybatis 多对多映射的小问题

本贴最后更新于 1883 天前,其中的信息可能已经时异事殊

问题

ORM 映射中,一对多,多对多是非常常见的方式。但是由于场景使用没有这么多,到今天我才发现这个问题。

配置映射关系,使字段与 DO 类解耦,方便维护。但是为了方便管理和控制,我就在一个 .java 文件中写了两个 VO 类。

@ToString @Setter(value = AccessLevel.PUBLIC) @Getter(value = AccessLevel.PUBLIC) @NoArgsConstructor @AllArgsConstructor @JsonInclude(value = JsonInclude.Include.NON_NULL) public class CategoryVO { private Long cid; private Long pid; private Integer level; private Integer sortNo; private String cname; private java.sql.Timestamp created; private java.sql.Timestamp updated; private List<Word> words; } @ToString @Setter(value = AccessLevel.PUBLIC) @Getter(value = AccessLevel.PUBLIC) @NoArgsConstructor @AllArgsConstructor @JsonInclude(value = JsonInclude.Include.NON_NULL) class Word { private Long id; private Long hotWordId; private Long categoryId; private String keyword; private Integer source; private Integer sortNo; private Integer isDeleted; private java.sql.Timestamp created; private java.sql.Timestamp updated; }

它们的关系也非常简单,可以直接看出,一个类目(分类)对应多组 HotWord(热词)。所以在 resultMap 中为了图快我就直接把属性一个个映射数据库的 column 了。

<resultMap id="result" type="{hape}.common.orm.model.vo.CategoryVO"> <!-- 相同的列名id --> <id column="id" jdbcType="INTEGER" property="cid"/> <result column="parent_id" jdbcType="INTEGER" property="pid"/> <result column="name" jdbcType="VARCHAR" property="cname"/> <result column="level" jdbcType="INTEGER" property="level"/> <result column="sort_no" jdbcType="INTEGER" property="sortNo"/> <collection property="words" ofType="{hape}.common.orm.model.vo.Word"> <!-- 相同的列名id --> <id column="id" jdbcType="INTEGER" property="id"/> <result column="hotword_id" jdbcType="INTEGER" property="hotWordId"/> <result column="category_id" jdbcType="INTEGER" property="categoryId"/> <result column="keyword" jdbcType="VARCHAR" property="keyword"/> <result column="source" jdbcType="TINYINT" property="source"/> <result column="sort_no" jdbcType="INTEGER" property="sortNo"/> <result column="is_deleted" jdbcType="TINYINT" property="isDeleted"/> <result column="created" jdbcType="TIMESTAMP" property="created"/> <result column="updated" jdbcType="TIMESTAMP" property="updated"/> </collection> </resultMap>

但是实际结果却并不是一个包含了多个子结构的对象,而是只有一个 words 对象,究其原因,原来是因为 colum 列名有重复,都存在一个相同的 id。

解决办法

你可以选择干掉其中一个 id,或者在使用 SQL 语句中加入别名加以区分,当字段较多的时候就会比较麻烦。我这里把字段都加上了 w_ 区分。

<select id="selectHotWordByCategoryId" parameterType="int" resultMap="result"> select <!-- 为什么要as "w_"呢?因为两张表的colum有相同的名称会出现问题,不然就只能查出子表的一条而不是多条了 --> t1.*, t2.id as w_id, t2.hotword_id as w_hotword_id, t2.keyword as w_keyword, t2.source as w_source, t2.sort_no as w_sort_no, t2.created as w_created, t2.updated as w_updated from xunfei_category t1 left join xunfei_hotword t2 on t1.id = t2.category_id where t1.id = #{id} and t2.is_deleted = 0 order by t2.sort_no, t2.updated desc </select>

然后在 resultMap 中修改成改后的别名就 OK 啦。

<resultMap id="result" type="{hape}.common.orm.model.vo.CategoryVO"> <id column="id" jdbcType="INTEGER" property="cid"/> <result column="parent_id" jdbcType="INTEGER" property="pid"/> <result column="name" jdbcType="VARCHAR" property="cname"/> <result column="level" jdbcType="INTEGER" property="level"/> <result column="sort_no" jdbcType="INTEGER" property="sortNo"/> <collection property="words" ofType="{hape}.common.orm.model.vo.Word"> <id column="w_id" jdbcType="INTEGER" property="id"/> <result column="w_hotword_id" jdbcType="INTEGER" property="hotWordId"/> <result column="w_category_id" jdbcType="INTEGER" property="categoryId"/> <result column="w_keyword" jdbcType="VARCHAR" property="keyword"/> <result column="w_source" jdbcType="TINYINT" property="source"/> <result column="w_sort_no" jdbcType="INTEGER" property="sortNo"/> <result column="w_is_deleted" jdbcType="TINYINT" property="isDeleted"/> <result column="w_created" jdbcType="TIMESTAMP" property="created"/> <result column="w_updated" jdbcType="TIMESTAMP" property="updated"/> </collection> </resultMap>
  • MyBatis

    MyBatis 本是 Apache 软件基金会 的一个开源项目 iBatis,2010 年这个项目由 Apache 软件基金会迁移到了 google code,并且改名为 MyBatis ,2013 年 11 月再次迁移到了 GitHub。

    171 引用 • 414 回帖 • 377 关注
  • 映射
    2 引用
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    345 引用 • 724 回帖
  • 多对多
    1 引用

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Vim

    Vim 是类 UNIX 系统文本编辑器 Vi 的加强版本,加入了更多特性来帮助编辑源代码。Vim 的部分增强功能包括文件比较(vimdiff)、语法高亮、全面的帮助系统、本地脚本(Vimscript)和便于选择的可视化模式。

    29 引用 • 66 回帖 • 1 关注
  • 服务

    提供一个服务绝不仅仅是简单的把硬件和软件累加在一起,它包括了服务的可靠性、服务的标准化、以及对服务的监控、维护、技术支持等。

    41 引用 • 24 回帖
  • abitmean

    有点意思就行了

    29 关注
  • 安装

    你若安好,便是晴天。

    132 引用 • 1184 回帖
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    226 引用 • 476 回帖
  • 安全

    安全永远都不是一个小问题。

    203 引用 • 816 回帖
  • Sillot

    Insights(注意当前设置 master 为默认分支)

    汐洛彖夲肜矩阵(Sillot T☳Converbenk Matrix),致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点。其中汐洛绞架(Sillot-Gibbet)基于自思源笔记(siyuan-note),前身是思源笔记汐洛版(更早是思源笔记汐洛分支),是智慧新录乄终端(多端融合,移动端优先)。

    主仓库地址:Hi-Windom/Sillot

    文档地址:sillot.db.sc.cn

    注意事项:

    1. ⚠️ 汐洛仍在早期开发阶段,尚不稳定
    2. ⚠️ 汐洛并非面向普通用户设计,使用前请了解风险
    3. ⚠️ 汐洛绞架基于思源笔记,开发者尽最大努力与思源笔记保持兼容,但无法实现 100% 兼容
    29 引用 • 25 回帖 • 108 关注
  • CodeMirror
    1 引用 • 2 回帖 • 148 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    345 引用 • 724 回帖
  • uTools

    uTools 是一个极简、插件化、跨平台的现代桌面软件。通过自由选配丰富的插件,打造你得心应手的工具集合。

    7 引用 • 27 回帖
  • 区块链

    区块链是分布式数据存储、点对点传输、共识机制、加密算法等计算机技术的新型应用模式。所谓共识机制是区块链系统中实现不同节点之间建立信任、获取权益的数学算法 。

    92 引用 • 752 回帖
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • Office

    Office 现已更名为 Microsoft 365. Microsoft 365 将高级 Office 应用(如 Word、Excel 和 PowerPoint)与 1 TB 的 OneDrive 云存储空间、高级安全性等结合在一起,可帮助你在任何设备上完成操作。

    5 引用 • 34 回帖
  • GraphQL

    GraphQL 是一个用于 API 的查询语言,是一个使用基于类型系统来执行查询的服务端运行时(类型系统由你的数据定义)。GraphQL 并没有和任何特定数据库或者存储引擎绑定,而是依靠你现有的代码和数据支撑。

    4 引用 • 3 回帖 • 2 关注
  • 服务器

    服务器,也称伺服器,是提供计算服务的设备。由于服务器需要响应服务请求,并进行处理,因此一般来说服务器应具备承担服务并且保障服务的能力。

    125 引用 • 585 回帖 • 1 关注
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    200 引用 • 120 回帖
  • 人工智能

    人工智能(Artificial Intelligence)是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门技术科学。

    152 引用 • 279 回帖
  • OpenCV
    15 引用 • 36 回帖 • 2 关注
  • Telegram

    Telegram 是一个非盈利性、基于云端的即时消息服务。它提供了支持各大操作系统平台的开源的客户端,也提供了很多强大的 APIs 给开发者创建自己的客户端和机器人。

    5 引用 • 35 回帖
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    211 引用 • 358 回帖
  • Typecho

    Typecho 是一款博客程序,它在 GPLv2 许可证下发行,基于 PHP 构建,可以运行在各种平台上,支持多种数据库(MySQL、PostgreSQL、SQLite)。

    12 引用 • 67 回帖 • 442 关注
  • C++

    C++ 是在 C 语言的基础上开发的一种通用编程语言,应用广泛。C++ 支持多种编程范式,面向对象编程、泛型编程和过程化编程。

    107 引用 • 153 回帖
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    171 引用 • 513 回帖
  • OneDrive
    2 引用
  • SVN

    SVN 是 Subversion 的简称,是一个开放源代码的版本控制系统,相较于 RCS、CVS,它采用了分支管理系统,它的设计目标就是取代 CVS。

    29 引用 • 98 回帖 • 686 关注
  • RabbitMQ

    RabbitMQ 是一个开源的 AMQP 实现,服务器端用 Erlang 语言编写,支持多种语言客户端,如:Python、Ruby、.NET、Java、C、PHP、ActionScript 等。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

    49 引用 • 60 回帖 • 355 关注
  • Ruby

    Ruby 是一种开源的面向对象程序设计的服务器端脚本语言,在 20 世纪 90 年代中期由日本的松本行弘(まつもとゆきひろ/Yukihiro Matsumoto)设计并开发。在 Ruby 社区,松本也被称为马茨(Matz)。

    7 引用 • 31 回帖 • 248 关注