Mybatis 多对多映射的小问题

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

问题

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 回帖 • 430 关注
  • 映射
    2 引用
  • 数据库

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

    330 引用 • 614 回帖
  • 多对多
    1 引用

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 28 关注
  • Vim

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

    27 引用 • 66 回帖
  • 开源

    Open Source, Open Mind, Open Sight, Open Future!

    396 引用 • 3416 回帖
  • ZeroNet

    ZeroNet 是一个基于比特币加密技术和 BT 网络技术的去中心化的、开放开源的网络和交流系统。

    1 引用 • 21 回帖 • 590 关注
  • 一些有用的避坑指南。

    69 引用 • 93 回帖
  • abitmean

    有点意思就行了

    22 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

    4 引用 • 7 回帖 • 3 关注
  • 尊园地产

    昆明尊园房地产经纪有限公司,即:Kunming Zunyuan Property Agency Company Limited(简称“尊园地产”)于 2007 年 6 月开始筹备,2007 年 8 月 18 日正式成立,注册资本 200 万元,公司性质为股份经纪有限公司,主营业务为:代租、代售、代办产权过户、办理银行按揭、担保、抵押、评估等。

    1 引用 • 22 回帖 • 684 关注
  • 阿里云

    阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

    89 引用 • 345 回帖
  • FreeMarker

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

    23 引用 • 20 回帖 • 426 关注
  • 智能合约

    智能合约(Smart contract)是一种旨在以信息化方式传播、验证或执行合同的计算机协议。智能合约允许在没有第三方的情况下进行可信交易,这些交易可追踪且不可逆转。智能合约概念于 1994 年由 Nick Szabo 首次提出。

    1 引用 • 11 回帖 • 8 关注
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖 • 93 关注
  • Solo

    Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    1425 引用 • 10043 回帖 • 469 关注
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    108 引用 • 54 回帖 • 1 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 55 关注
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖
  • 负能量

    上帝为你关上了一扇门,然后就去睡觉了....努力不一定能成功,但不努力一定很轻松 (° ー °〃)

    85 引用 • 1201 回帖 • 449 关注
  • wolai

    我来 wolai:不仅仅是未来的云端笔记!

    1 引用 • 11 回帖 • 2 关注
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 403 关注
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 350 关注
  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 460 关注
  • OAuth

    OAuth 协议为用户资源的授权提供了一个安全的、开放而又简易的标准。与以往的授权方式不同之处是 oAuth 的授权不会使第三方触及到用户的帐号信息(如用户名与密码),即第三方无需使用用户的用户名与密码就可以申请获得该用户资源的授权,因此 oAuth 是安全的。oAuth 是 Open Authorization 的简写。

    36 引用 • 103 回帖 • 9 关注
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    25 引用 • 191 回帖 • 21 关注
  • 导航

    各种网址链接、内容导航。

    37 引用 • 168 回帖
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    60 引用 • 287 回帖
  • 爬虫

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

    106 引用 • 275 回帖
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    163 引用 • 1450 回帖