范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文
国学影视

MyBatisPlus中如何使用ResultMap

  作者:字节飞扬
  原文链接:https://www.cnblogs.com/bytesfly/p/resultmap-in-mybatis-plus.html
  MyBatis-Plus (简称 MP )是一个MyBatis 的增强工具,在MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。
  MyBatis-Plus 对MyBatis 基本零侵入,完全可以与MyBatis 混合使用,这点很赞。
  在涉及到关系型数据库增删查改的业务时,我比较喜欢用 MyBatis-Plus ,开发效率极高。具体的使用可以参考官网,或者自己上手摸索感受一下。
  下面简单总结一下在 MyBatis-Plus 中如何使用ResultMap 。问题说明#
  先看个例子:
  有如下两张表: create table tb_book (     id     bigint primary key,     name   varchar(32),     author varchar(20) );  create table tb_hero (     id      bigint primary key,     name    varchar(32),     age     int,     skill   varchar(32),     bid bigint );
  其中, tb_hero 中的bid 关联tb_book 表的id 。
  下面先看 Hero 实体类的代码,如下:import com.baomidou.mybatisplus.annotation.TableField; import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableName; import com.fasterxml.jackson.annotation.JsonInclude; import lombok.Getter; import lombok.NoArgsConstructor; import lombok.Setter;  @Getter @Setter @NoArgsConstructor @TableName("tb_hero") @JsonInclude(JsonInclude.Include.NON_NULL) public class Hero {      @TableId("id")     private Long id;      @TableField(value = "name", keepGlobalFormat = true)     private String name;      @TableField(value = "age", keepGlobalFormat = true)     private Integer age;      @TableField(value = "skill", keepGlobalFormat = true)     private String skill;      @TableField(value = "bid", keepGlobalFormat = true)     private Long bookId;      // *********************************     // 数据库表中不存在以下字段(表join时会用到)     // *********************************      @TableField(value = "book_name", exist = false)     private String bookName;      @TableField(value = "author", exist = false)     private String author; }
  注意了,我特地把 tb_hero 表中的bid 字段映射成实体类Hero 中的bookId 属性。测试 BaseMapper 中内置的insert() 方法或者IService 中的save() 方法
  MyBatis-Plus 打印出的SQL 为:==> Preparing: INSERT INTO tb_hero ( id, "name", "age", "skill", "bid" ) VALUES ( ?, ?, ?, ?, ? ) ==> Parameters: 1589788935356416(Long), 阿飞(String), 18(Integer), 天下第一快剑(String), 1(Long)
  没毛病,  MyBatis-Plus 会根据@TableField 指定的映射关系,生成对应的SQL 。测试 BaseMapper 中内置的selectById() 方法或者IService 中的getById() 方法
  MyBatis-Plus 打印出的SQL 为:==> Preparing: SELECT id,"name","age","skill","bid" AS bookId FROM tb_hero WHERE id=? ==> Parameters: 1(Long)
  也没毛病,可以看到生成的 SELECT 中把bid 做了别名bookId 。测试自己写的SQL
  比如现在我想连接 tb_hero 与tb_book 这两张表,如下:@Mapper @Repository public interface HeroMapper extends BaseMapper {     @Select({"SELECT tb_hero.*, tb_book.name as book_name, tb_book.author" +             " FROM tb_hero" +             " LEFT JOIN tb_book" +             " ON tb_hero.bid = tb_book.id" +             " ${ew.customSqlSegment}"})     IPage pageQueryHero(@Param(Constants.WRAPPER) Wrapper queryWrapper,                               Page page); }
  查询 MyBatis-Plus 打印出的SQL 为:==> Preparing: SELECT tb_hero.*, tb_book.name AS book_name, tb_book.author FROM tb_hero LEFT JOIN tb_book ON tb_hero.bid = tb_book.id WHERE ("bid" = ?) ORDER BY id ASC LIMIT ? OFFSET ? ==> Parameters: 2(Long), 1(Long), 1(Long)
  SQL没啥问题,过滤与分页也都正常,但是此时你会发现 bookId 属性为null ,如下:
  为什么呢?
  调用 BaseMapper 中内置的selectById() 方法并没有出现这种情况啊???
  回过头来再对比一下在 HeroMapper 中自己定义的查询与MyBatis-Plus 自带的selectById() 有啥不同,还记得上面的刚刚的测试吗,生成的SQL有啥不同?
  原来, MyBatis-Plus 为BaseMapper 中内置的方法生成SQL时,会把SELECT 子句中bid 做别名bookId ,而自己写的查询MyBatis-Plus 并不会帮你修改SELECT 子句,也就导致bookId 属性为null 。解决方法#方案一:表中的字段与实体类的属性严格保持一致(字段有下划线则属性用驼峰表示)
  在这里就是 tb_hero 表中的bid 字段映射成实体类Hero 中的bid 属性。这样当然可以解决问题,但不是本篇讲的重点。方案二:把自己写的 SQL 中bid 做别名bookId 方案三:使用 @ResultMap ,这是此篇的重点
  在 @TableName 设置autoResultMap = true @TableName(value = "tb_hero", autoResultMap = true) public class Hero {      }
  然后在自定义查询中添加 @ResultMap 注解,如下:import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Param; import org.apache.ibatis.annotations.ResultMap; import org.apache.ibatis.annotations.Select; import org.springframework.stereotype.Repository;  @Mapper @Repository public interface HeroMapper extends BaseMapper {     @ResultMap("mybatis-plus_Hero")     @Select({"SELECT tb_hero.*, tb_book.name as book_name, tb_book.author" +             " FROM tb_hero" +             " LEFT JOIN tb_book" +             " ON tb_hero.bid = tb_book.id" +             " ${ew.customSqlSegment}"})     IPage pageQueryHero(@Param(Constants.WRAPPER) Wrapper queryWrapper,                               Page page); }
  这样,也能解决问题。
  下面简单看下源码, @ResultMap("mybatis-plus_实体类名") 怎么来的。
  详情见:  com.baomidou.mybatisplus.core.metadata.TableInfo#initResultMapIfNeed() /**  * 自动构建 resultMap 并注入(如果条件符合的话)  */ void initResultMapIfNeed() {     if (autoInitResultMap && null == resultMap) {         String id = currentNamespace + DOT + MYBATIS_PLUS + UNDERSCORE + entityType.getSimpleName();         List resultMappings = new ArrayList<>();         if (havePK()) {             ResultMapping idMapping = new ResultMapping.Builder(configuration, keyProperty, StringUtils.getTargetColumn(keyColumn), keyType)                 .flags(Collections.singletonList(ResultFlag.ID)).build();             resultMappings.add(idMapping);         }         if (CollectionUtils.isNotEmpty(fieldList)) {             fieldList.forEach(i -> resultMappings.add(i.getResultMapping(configuration)));         }         ResultMap resultMap = new ResultMap.Builder(configuration, id, entityType, resultMappings).build();         configuration.addResultMap(resultMap);         this.resultMap = id;     } }
  注意看上面的字符串 id 的构成,你应该可以明白。
  思考: 这种方式的 ResultMap 默认是强绑在一个@TableName 上的,如果是某个聚合查询或者查询的结果并非对应一个真实的表怎么办呢?有没有更优雅的方式?自定义@AutoResultMap注解#
  基于上面的思考,我做了下面简单的实现: 自定义@AutoResultMap注解 import java.lang.annotation.*;  /**  * 使用@AutoResultMap注解的实体类  * 自动生成{auto.mybatis-plus_类名}为id的resultMap  * {@link MybatisPlusConfig#initAutoResultMap()}  */ @Documented @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.TYPE) public @interface AutoResultMap {  } 启动时扫描@AutoResultMap注解的实体类 package com.bytesfly.mybatis.config;  import cn.hutool.core.util.ClassUtil; import cn.hutool.core.util.ReflectUtil; import com.baomidou.mybatisplus.annotation.DbType; import com.baomidou.mybatisplus.core.metadata.TableInfo; import com.baomidou.mybatisplus.core.metadata.TableInfoHelper; import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor; import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor; import com.baomidou.mybatisplus.extension.toolkit.JdbcUtils; import com.bytesfly.mybatis.annotation.AutoResultMap; import lombok.extern.slf4j.Slf4j; import org.apache.ibatis.builder.MapperBuilderAssistant; import org.mybatis.spring.SqlSessionTemplate; import org.mybatis.spring.annotation.MapperScan; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.transaction.annotation.EnableTransactionManagement;  import javax.annotation.PostConstruct; import java.util.Set;  /**  * 可添加一些插件  */ @Configuration @EnableTransactionManagement(proxyTargetClass = true) @MapperScan(basePackages = "com.bytesfly.mybatis.mapper") @Slf4j public class MybatisPlusConfig {      @Autowired     private SqlSessionTemplate sqlSessionTemplate;      /**      * 分页插件(根据jdbcUrl识别出数据库类型, 自动选择适合该方言的分页插件)      * 相关使用说明: https://baomidou.com/guide/page.html      */     @Bean     public MybatisPlusInterceptor mybatisPlusInterceptor(DataSourceProperties dataSourceProperties) {          String jdbcUrl = dataSourceProperties.getUrl();         DbType dbType = JdbcUtils.getDbType(jdbcUrl);          MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();         interceptor.addInnerInterceptor(new PaginationInnerInterceptor(dbType));         return interceptor;     }      /**      * @AutoResultMap注解的实体类自动构建resultMap并注入      */     @PostConstruct     public void initAutoResultMap() {         try {             log.info("--- start register @AutoResultMap ---");              String namespace = "auto";              String packageName = "com.bytesfly.mybatis.model.db.resultmap";             Set> classes = ClassUtil.scanPackageByAnnotation(packageName, AutoResultMap.class);              org.apache.ibatis.session.Configuration configuration = sqlSessionTemplate.getConfiguration();              for (Class clazz : classes) {                 MapperBuilderAssistant assistant = new MapperBuilderAssistant(configuration, "");                 assistant.setCurrentNamespace(namespace);                 TableInfo tableInfo = TableInfoHelper.initTableInfo(assistant, clazz);                  if (!tableInfo.isAutoInitResultMap()) {                     // 设置 tableInfo的autoInitResultMap属性 为 true                     ReflectUtil.setFieldValue(tableInfo, "autoInitResultMap", true);                     // 调用 tableInfo#initResultMapIfNeed() 方法,自动构建 resultMap 并注入                     ReflectUtil.invoke(tableInfo, "initResultMapIfNeed");                 }             }              log.info("--- finish register @AutoResultMap ---");         } catch (Throwable e) {             log.error("initAutoResultMap error", e);             System.exit(1);         }     } }
  关键代码其实没有几行,耐心看下应该不难懂。 使用@AutoResultMap注解
  还是用例子来说明更直观。
  下面是一个聚合查询: @Mapper @Repository public interface BookMapper extends BaseMapper {      @ResultMap("auto.mybatis-plus_BookAgg")     @Select({"SELECT tb_book.id, max(tb_book.name) as name, array_agg(distinct tb_hero.id order by tb_hero.id asc) as hero_ids" +             " FROM tb_hero" +             " INNER JOIN tb_book" +             " ON tb_hero.bid = tb_book.id" +             " GROUP BY tb_book.id"})     List agg(); }
  其中 BookAgg 的定义如下,在实体类上使用了@AutoResultMap 注解:@Getter @Setter @NoArgsConstructor @AutoResultMap public class BookAgg {      @TableId("id")     private Long bookId;      @TableField("name")     private String bookName;      @TableField("hero_ids")     private Object heroIds; }

别搞错了,失业金和失业补助金不是一码事,不建议领失业补助金别搞错了,失业金和失业补助金不是一码事,不建议领失业补助金。失业金和失业补助金从字面上看,只有两个字的差别,现实中不少人搞不清二者具有什么关系。从二者的历史发展来看,失业金是一个老(财经行情)纽约金价19日下跌新华社芝加哥10月19日电(记者徐静)纽约商品交易所黄金期货市场交投最活跃的12月黄金期价19日比前一交易日下跌21。6美元,收于每盎司1634。2美元,跌幅为1。3。市场分析人士美元指数目前以震荡为主,贵金属方面后市弱势行情偏空在昨天的美元指数行情分析里,个人认为美元指数会接下来在没有消息面的影响下会运行在111112区间,这一结论主要依据是根据前期的基本面和最近英格兰亨特公布的财税政策得到的。如果还有粉遭恐慌性抛售!业绩销售不算差,金地何以至此?摘要市场如同惊弓之鸟,真是一个有趣的时代(欢迎关注杠杆地产)撰文杆姐编辑雯雯诡异了,一个能提前偿债,年内再无到期公开债务的房企,债券近日却遭遇恐慌性抛售。继2022年10月18日1比亚迪预计前三季净利润大涨2。7倍10月17日,比亚迪发布业绩预告称,预计2022年前三季度实现净利润91亿元至95亿元,同比大涨2。7倍以上。表现更出色的是扣非净利润,前三季度预计达到81亿元至88亿元,同比大增中航电子获控股股东出资4000万增持推进重组中航系多家公司拟参与定增长江商报奔腾新闻记者蔡嘉在推进重大资产重组的关键时间点,中航电子(600372。SH)获得控股股东增持。10月17日晚间,中航电子披露,公司控股股东中国航空科技工业股份有限公司(以阿科力前三季最高预盈1。1亿连续三年满产满销拟10。5亿扩产长江商报记者张佩长江商报消息连续三年实现满产满销,阿科力(603722。SH)拟大举扩产增加产能。10月17日晚间,阿科力抛出新一轮投建计划。公司近日与潜江江汉管委会签署投资协议,部分事业单位的退休人员,70岁之后,可领取的养老金会变少?守护银龄世界养老金已然成为绝大多数退休老人,失去一定工作能力后的唯一经济来源,俗话说经济基础决定上层建筑,一直以来养老金的多少,都是老人们最为关注的话题之一。相较于城镇居民来说,城我们如何看待孩子的边界?边界是什么呢?如果在网上搜索结果,我们会看到一条解释。边界指的是,地区和地区之间的界线。或许可以想象成将复数的不同的事物与空间隔开的一条线。将边界放在人的社交言行身上,似乎很好理解茂名诺亚舟幼儿园参观消防大队活动精彩回顾茂名诺亚舟幼儿园参观消防大队活动精彩回顾哪有什么岁月静好,不过是有人替你负重前行为了增强孩子的消防安全意识,使他们从小接受消防知识教育,不断深入对消防知识的了解,增强自防自救能力。鲁豫瘦到病态,看脖子像六七十岁的老太太,穿起球的套装略寒酸鲁豫现在看起来真的和67岁的老太太没区别。前一段时间有人看到她发的照片整个人已经瘦到了病态。尤其是看她脖子的时候,皱纹特别的明显根根分明。再加上她穿的衣服又是起球的套装,感觉她整个董洁对黄梅莹说您和我一起穿旗袍演母女吧!如果让你在所有演员中选出两个人演母女,你会选谁呢?另外,还有一个要求,那就是这两位还都必须是穿旗袍都好看的女神。你首先想到的是谁?应该是黄梅莹和董洁吧!那说明你的眼光和导演的眼光差同样是国模,刘雯奚梦瑶何穗雎晓雯的差距有多大?文丨水星图丨来源于网络前几天的VogueWorld展览开幕之夜,算是内娱今年为数不多的高质量红毯了。像这种时尚修罗场,女星们自然个个拿出最好状态。作为现场的唯一大花,李冰冰更是格外WTT澳门冠军赛直播20日赛程,樊振东孙颖莎王曼昱首秀头条创作挑战赛2022年10月20日WTT澳门冠军赛进入第二个比赛日,将继续进行16场男单女单116决赛。从上午1000一直打到晚上2200,12个小时不间断的精彩比赛,咪咕视频行冷静!阿森纳球星还没准备好去皇家马德加布里埃尔马蒂内利已经被告知他还没有足够好为皇家马德里效力,但是阿森纳必须和他签订一份新的合同。枪手在2019年成功签下了巴西球队伊图亚诺的马蒂内利,只花费了他们600万欧元。这名波巴回来了!尤文图斯中场在膝盖受伤后恢复部分训练尤文图斯Z博中场保罗波格巴从膝伤中恢复,目前已恢复部分训练。Pogba季前赛膝盖受伤9月做过手术现在又开始训练了由于膝盖受伤,博格巴在经历了长时间的停赛后,于周二回到尤文图斯训练。三冠王将帅对决,穆里尼奥再给斯坦科维奇上一课?阵容桑普(442)奥代罗贝雷申斯基费拉里科利奥杰洛萨比利林孔比利亚尔久里西奇卡普托加比亚迪尼主教练斯坦科维奇罗马(3421)帕特里西奥曼奇尼斯莫林伊巴涅斯扎勒夫斯基克里斯坦特马蒂奇1019A阿伟足球解析英超纽卡斯尔vs埃弗顿纽卡斯尔埃弗顿202210200230纽卡斯尔近期状态非常稳定,当下三胜六平一负的战绩,位列联赛第六位。球队的联赛竞争力似乎不是很足,但其实此球队近期伤病情况是较为严重的,能够在这逼出最强龙队!侯英超宇田招惹马龙干嘛?WTT澳门冠军赛赛程WTT澳门冠军赛首轮马龙3比1战胜宇田幸矢,晋级1男队6强!四局比分马龙1191315113112宇田幸矢。这场比赛,平平无奇,但是宇田幸矢的第二局取胜,逼出了最强龙队马龙!咪咕体2米21的中锋,场均4。7分4。3个篮板,他为何能拿顶薪?一个身高2米21体重125公斤的球员,得分不行,抢篮板不行,助攻不行,抢断和盖帽更加不行,除了身高和体重优势以外没有其他亮点,却能拿到CBA的顶薪合同,这个球员就是张兆旭。天津队本张常宁毁掉大好前途?不及时回归丧失位置,蔡斌指望不上选择放弃头条创作挑战赛张常宁不及时回归付出代价!近日2022年的女排世锦赛已经结束,塞尔维亚连续两届赛事夺得最终的冠军,展现了自己强大的实力,而中国女排在这次赛事中最终取得了第6名的成绩,写心得体会总结报告怎能少得了这些金句?上下同欲者胜,以上率下者强!凝心聚力创伟业,万众一心向未来。又踏层峰辟新天,更扬云帆立潮头。伟大梦想不是等得来喊得来的,而是拼出来干出来的。利剑千锤成器,精铁百炼成钢。奋楫笃行,臻
向往的生活云南最大!阳宗海500亩芍药牡丹盛放来源昆明日报掌上春城掌上春城讯4月19日,2023年阳宗海第二届文化旅游节暨首届梁王山牡丹芍药文化节新闻发布会在滇香国色植物园召开。目前,云南最大的芍药牡丹观赏基地滇香国色植物园5轩辕故里甘肃清水10万亩野生丁香绽放美丽经济中新网甘肃新闻4月18日电(崔琳)近日,甘肃省天水市清水县2023年丁香文化旅游周宣传推介会举行,推介了以旅为媒擦亮地域品牌以文为魂赋能融合发展以农为基绽放美丽经济以康为根撬动健康听阿佤唱新歌来源人民网人民日报海外版那天,我们从云南省腾冲市区出发,沿着通畅的乡村公路,前往司莫拉佤族村。司莫拉,佤语意为幸福的地方。还未进村,就听见村村寨寨哎,打起鼓,敲起锣,阿佤唱新歌的悠如梦似幻凤凰古城离开湘西凤凰古城多时了,一直想写点文字,记录一下我的感受,但是坐到电脑前却不知写什么好。我不知怎么表述自已对这座边远小城的印象,关于它,夜晚和清晨竟是如此不同,现代和古典又是如此的梁思成盛赞的正定隆兴寺,到底有何绝妙?这么近那么美周末到河北河北文旅看图识景奋进新时代美丽石家庄隆兴寺位于河北省正定县,始建于隋,初名龙藏寺,距今已有1600余年历史。北宋时大规模扩建,后虽经金元明清历代重修,但仍保持穴位密码之印堂穴来源印堂穴,最早可追溯至黄帝内经,素问刺疟篇载先头痛及重者,先刺头上及两额两眉间出血,当时有定位及主治病症而无穴名,灵枢五色篇中称之为阙者,眉间也。不少医书又以眉心为其代称。至唐代襄垣县下良镇寨上村首届桃花节以花为媒以旅兴农助力乡村振兴又是一年春风起,桃花朵朵烂漫开。4月16日,襄垣县下良镇寨上村首届桃花节开幕。该村以桃花节为主题,打造集踏青赏花亲子活动休闲娱乐为一体的户外游园活动,游客以花为媒,以节会友,踏着和曾是金戈铁马的战场,逝去的战神化作擎天柱,捍卫着心中的圣殿如若把七彩丹霞比作为风情万种的艳后,那么冰沟丹霞就是罗马帝国的战神。两者仅仅相距10公里,又同为丹霞地貌却风格迥异,一个艳丽,一个雄浑,或许应验了那句话天造的一对,地设的一双!不望跟着唐诗宋词去旅游诗路云帆踏歌行朝阳的手撩开了夜幕,也撩动了诗人的心弦。在浙东重重的青山间,一席白帆急切地向南。唐朝开元十五年前后,孟浩然自越州出发,循着曹娥江往天台山,写下了这首舟中晓望。曹娥江是浙东名水,青山乡村振兴文明赋能染太行中国网中国发展门户网讯山西左权县,拥有众多人文自然旅游资源。近年来推动的全域旅游工作,使处于太行山腹地的左权县桐峪镇上武村旅游资源得到了深度开发,乡村旅游广受欢迎。春天里,百亩桃园红色文旅黔贵乡村游首发团,带你领略少数民族风土人情黔贵乡村游本次行程路线共五天四夜4月10日广州出发,高铁往返包括贵州安顺镇宁高荡村4A景区黄果树大瀑布,青岩古镇荔波大小七孔,西江千户苗寨带听众领略了少数民族风土人情听众表示满意并