Mybatis 多对多映射的小问题

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

问题

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。

    170 引用 • 414 回帖 • 431 关注
  • 映射
    2 引用
  • 数据库

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

    330 引用 • 614 回帖 • 2 关注
  • 多对多
    1 引用

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • 强迫症

    强迫症(OCD)属于焦虑障碍的一种类型,是一组以强迫思维和强迫行为为主要临床表现的神经精神疾病,其特点为有意识的强迫和反强迫并存,一些毫无意义、甚至违背自己意愿的想法或冲动反反复复侵入患者的日常生活。

    15 引用 • 161 回帖 • 1 关注
  • 音乐

    你听到信仰的声音了么?

    59 引用 • 509 回帖
  • 996
    13 引用 • 200 回帖 • 1 关注
  • QQ

    1999 年 2 月腾讯正式推出“腾讯 QQ”,在线用户由 1999 年的 2 人(马化腾和张志东)到现在已经发展到上亿用户了,在线人数超过一亿,是目前使用最广泛的聊天软件之一。

    45 引用 • 557 回帖 • 227 关注
  • MongoDB

    MongoDB(来自于英文单词“Humongous”,中文含义为“庞大”)是一个基于分布式文件存储的数据库,由 C++ 语言编写。旨在为应用提供可扩展的高性能数据存储解决方案。MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的。它支持的数据结构非常松散,是类似 JSON 的 BSON 格式,因此可以存储比较复杂的数据类型。

    90 引用 • 59 回帖 • 2 关注
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    7 引用 • 26 回帖 • 3 关注
  • RIP

    愿逝者安息!

    8 引用 • 92 回帖 • 293 关注
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖 • 1 关注
  • FreeMarker

    FreeMarker 是一款好用且功能强大的 Java 模版引擎。

    23 引用 • 20 回帖 • 427 关注
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖 • 2 关注
  • CodeMirror
    1 引用 • 2 回帖 • 114 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 547 关注
  • 反馈

    Communication channel for makers and users.

    123 引用 • 906 回帖 • 191 关注
  • CSDN

    CSDN (Chinese Software Developer Network) 创立于 1999 年,是中国的 IT 社区和服务平台,为中国的软件开发者和 IT 从业者提供知识传播、职业发展、软件开发等全生命周期服务,满足他们在职业发展中学习及共享知识和信息、建立职业发展社交圈、通过软件开发实现技术商业化等刚性需求。

    14 引用 • 155 回帖
  • ActiveMQ

    ActiveMQ 是 Apache 旗下的一款开源消息总线系统,它完整实现了 JMS 规范,是一个企业级的消息中间件。

    19 引用 • 13 回帖 • 628 关注
  • Swift

    Swift 是苹果于 2014 年 WWDC(苹果开发者大会)发布的开发语言,可与 Objective-C 共同运行于 Mac OS 和 iOS 平台,用于搭建基于苹果平台的应用程序。

    34 引用 • 37 回帖 • 495 关注
  • Laravel

    Laravel 是一套简洁、优雅的 PHP Web 开发框架。它采用 MVC 设计,是一款崇尚开发效率的全栈框架。

    19 引用 • 23 回帖 • 682 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    51 引用 • 37 回帖
  • Flutter

    Flutter 是谷歌的移动 UI 框架,可以快速在 iOS 和 Android 上构建高质量的原生用户界面。 Flutter 可以与现有的代码一起工作,它正在被越来越多的开发者和组织使用,并且 Flutter 是完全免费、开源的。

    39 引用 • 92 回帖 • 7 关注
  • BAE

    百度应用引擎(Baidu App Engine)提供了 PHP、Java、Python 的执行环境,以及云存储、消息服务、云数据库等全面的云服务。它可以让开发者实现自动地部署和管理应用,并且提供动态扩容和负载均衡的运行环境,让开发者不用考虑高成本的运维工作,只需专注于业务逻辑,大大降低了开发者学习和迁移的成本。

    19 引用 • 75 回帖 • 621 关注
  • 链书

    链书(Chainbook)是 B3log 开源社区提供的区块链纸质书交易平台,通过 B3T 实现共享激励与价值链。可将你的闲置书籍上架到链书,我们共同构建这个全新的交易平台,让闲置书籍继续发挥它的价值。

    链书社

    链书目前已经下线,也许以后还有计划重制上线。

    14 引用 • 257 回帖
  • 房星科技

    房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

    6 引用 • 141 回帖 • 553 关注
  • Gzip

    gzip (GNU zip)是 GNU 自由软件的文件压缩程序。我们在 Linux 中经常会用到后缀为 .gz 的文件,它们就是 Gzip 格式的。现今已经成为互联网上使用非常普遍的一种数据压缩格式,或者说一种文件格式。

    9 引用 • 12 回帖 • 111 关注
  • Sillot

    Sillot (汐洛)孵化自思源笔记,致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点
    Github 地址:https://github.com/Hi-Windom/Sillot

    14 引用 • 4 回帖 • 26 关注
  • 自由行
    1 关注
  • OpenStack

    OpenStack 是一个云操作系统,通过数据中心可控制大型的计算、存储、网络等资源池。所有的管理通过前端界面管理员就可以完成,同样也可以通过 Web 接口让最终用户部署资源。

    10 引用 • 9 关注
  • GitHub

    GitHub 于 2008 年上线,目前,除了 Git 代码仓库托管及基本的 Web 管理界面以外,还提供了订阅、讨论组、文本渲染、在线文件编辑器、协作图谱(报表)、代码片段分享(Gist)等功能。正因为这些功能所提供的便利,又经过长期的积累,GitHub 的用户活跃度很高,在开源世界里享有深远的声望,并形成了社交化编程文化(Social Coding)。

    207 引用 • 2031 回帖