解决问题:n+1问题,之前我的习惯是拿到单表里面的数据,然后遍历,再拿到一个与其他表对应的逻辑上的外键,然后for循环去查找其他表的数据(原因是数据量小,没有在意,还有主要是不想建外键,你知道的,外键是很麻烦的,虽然有利于查询,但是增删改确实很让人头疼),这样做也能达到效果,但是效率低,访问数据库的次数也太多了,假设我查询出了1000条数据,我要用他里面的逻辑外键去查找其他表1000次,就意味着访问数据1000次,这样做还会遇到一些问题,若当前的逻辑外键查找在对应的表里面没有数据,就会抛出异常,从程序出错

  例子,我要返回前端一个封装好的实体模型,里面包含了多个不在同一个数据表的字段,这就涉及到了多表查询,我把需要返回前端的字段封装为一个DTO

  步骤:

    1:构建DTO

package com.steak.system.pojo.dto;

public class ApplyDTO {
private Integer applyId; //申请书的ID 属于申请表(sys_apply)
private String selfIntroduction; //自我介绍 属于申请表(sys_apply)
private String applyTime; //申请时间 属于申请表(sys_apply)
private String userName; //申请人,属于用户表(sys_user)
private String collegeName; //二级学院名称,属于二级学院表(sys_college)
private String recruitName; //招聘书标题 , 属于招聘表(sys_recruit)
private String status; //申请状态 属于申请表(sys_apply)
   省去setter,getter
}   从上面可以看出上面的字段涉及了四个表

    2:构建mybatis Mapper.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.steak.system.mapper.ApplyDOMapper">
<resultMap id="ApplyDTO" type="com.steak.system.pojo.dto.ApplyDTO">
<id column="apply_id" jdbcType="INTEGER" property="applyId" />
<result column="self_introduction" jdbcType="VARCHAR" property="selfIntroduction" />
<result column="user_name" jdbcType="VARCHAR" property="userName" />
<result column="apply_time" jdbcType="TIMESTAMP" property="applyTime" />
<result column="college_name" jdbcType="VARCHAR" property="collegeName" />
<result column="status" jdbcType="INTEGER" property="status" />
<result column="recruit_name" jdbcType="VARCHAR" property="recruitName" />
</resultMap> <select id="getApplyDTO" parameterType="java.lang.String" resultMap="ApplyDTO">
select a.apply_id,a.self_introduction,a.apply_time,u.user_name,c.college_name,a.status,r.recruit_name
from sys_recruit r,sys_apply a,sys_user u,sys_college c
where a.recruit_id = r.recruit_id and a.apply_people_id = u.user_id and r.apply_depatment = c.college_id
and release_people_id = #{userId,jdbcType=VARCHAR}
</select>
</mapper>   上面的resultMap 的 id="ApplyDTO"和下面的查询语句的resultMap要对应,resultMap算是mybatis中最强大的了,type="com.steak.system.pojo.dto.ApplyDTO"
  是我自定义的DTO(再次强调,DTO并不是规则的实体,如果不是很理解,可以去了解一下DTO) property的字段应和DTO里面的字段对应

    3:创建接口Mapper

List<ApplyDTO> getApplyDTO(String userId);
getApplyDTO要和xml里面的查询语句的id对应

    4:返回结果

List<ApplyDTO> list = applyDOMapper.getApplyDTO(userId);

在service层调用Mapper的接口,就能返回对应的结果

    

    额外:实体之间的转换(少写代码)

      如:我封装的DTO里面有一个字段int类型的字段status,一般在数据库里面我们用数字来表示,但是返回前台我们应该用一个别人能够理解的字段来表示(如待审核,已审核),由此我们需要定义一个VO(返回前端的模型,里面的字段要和DTO里面所要转换的的字段相同,注意,是需要转换的,我也可以在VO里面定义其他的,DTO转换的时候只转换VO里面和自己对应的),然后通过set方式将值转换到VO里面,数据字段少其实可以手动set,如果有50个字段,一个字段一个字段的去set(实际上我只需要转换一个status字段就OK,当然也可以交给前端来处理,但是有时候前端并不知道代表什么,所以我们尽量给前端解析好,别人就不用去猜测了),费力费时,代码还不优雅,所以推荐使用一些实体的转换工具,我使用的是Apache的BeanUtils,里面有转换的方法,如我将DTO转换为VO,如下便可

  BeanUtils.copyProperties(VO,DTO);

 这样的话我就将所有数据转到VO里面了,我们在VO里面的status字段设置为String,然后取DTO里面的status字段出来进行语义化成一个可理解的字段,为什么不取VO里面的,要取DTO呢,前面已经说了只能转换和自己对应的,因为DTO里面的status为int类型,而VO里面的status为String,所以转过来VO里面的status为null,所以需要取DTO里面的status,然后赋值给VO里面的status,如:1=待审核 2=已审核

  本文提到的VO,DTO模型并不是真正意义上的实体,它不像DO这种是直接和数据库一一对应的实体,它是可拓展的,可变的,如果不理解,可以百度一下,另外本文纯属CRUD,不喜勿喷

Mybatis ResultMap多表映射DTO的更多相关文章

  1. mybatis 使用resultMap实现表间关联

    AutoMapping auto mapping,直译过来就是自动映射,工作原理大概如下: 假设我们有一张表,表名为person,包含id,name,age,addr这4个字段 mysql> d ...

  2. MyBatis多表映射demo

    三个实体类,作者.文章和评论. public class Author { private int id; private String username; private String nickna ...

  3. 深入浅出Mybatis系列八-mapper映射文件配置之select、resultMap

    注:本文转载自南轲梦 注:博主 Chloneda:个人博客 | 博客园 | Github | Gitee | 知乎 上篇<深入浅出Mybatis系列(七)---mapper映射文件配置之inse ...

  4. MyBatis快速入门(1):搭建环境和单表映射

    一.MyBatis简介    一说起对象关系映射框架,大家第一时间想到的肯定是Hibernate.Hibernate作为一个著名的框架,功能十分强大.我们只需要配置好实体类和数据表之间的关系,Hibe ...

  5. mybatis框架-resultMap的自动映射级别-partial 和full的探讨

    现在我们做一个小实验,输出一下上一个案例中没有匹配的属性,注意哦,现在user类中是有内部嵌套的复杂数据类型的 运行结果: 注意到:现在居然连userPassword都打印不出来了,原因就是user类 ...

  6. 【mybatis深度历险系列】mybatis中的高级映射一对一、一对多、多对多

    学习hibernate的时候,小编已经接触多各种映射,mybatis中映射有到底是如何运转的,今天这篇博文,小编主要来简单的介绍一下mybatis中的高级映射,包括一对一.一对多.多对多,希望多有需要 ...

  7. mybatis ResultMap详解

    前言 MyBatis是基于“数据库结构不可控”的思想建立的,也就是我们希望数据库遵循第三范式或BCNF,但实际事与愿违,那么结果集映射就是MyBatis为我们提供这种理想与现实间转换的手段了,而res ...

  8. mybatis 动态添加表,查看表,添加数据

    1.动态添加表 mapper int dropExistTable(@Param("tableName") String tableName);//自动创建数据表 映射文件 < ...

  9. mybatis 查询优化主子表查询之association和collection

    很多开发人员之所以编写出低效的应用,有一大原因是并不理解怎样编写高效的SQL.以订单查询为例,我们经常需要查询某个用户的订单以及订单明细,并且以树形方式展现如下: 对于这种性质的功能,很多开发人员的做 ...

随机推荐

  1. Linux学习笔记-第10天 特殊的交换分区

    关键词,分区.mkswap swapon .uquota,RAID,/etc/fstab 此章开始,难度有些提升.不过还好自己有点基础.

  2. 【正则】day01

    正则表达式一.概述    验证    网络爬虫. 概念:    具有语法格式的字符串. 函数    PCRE    1.perl语言正则语法兼容.(java c)    2.速度快,效率高.    P ...

  3. Educational Codeforces Round 63 (Rated for Div. 2) D dp(最大连续子序列)

    https://codeforces.com/contest/1155/problem/D 题意 一个n个数的数组\(a[i]\),可以选择连续的一段乘x,求最大连续子序列的值 题解 错误思路:贪心, ...

  4. 【2019.8.11上午 慈溪模拟赛 T3】欢迎回来(back)(设阈值+莫队)

    设阈值 考虑对于询问的\(d\)设阈值进行分别处理. 对于\(d\le\sqrt{max\ d}\)的询问,我们可以\(O(n\sqrt{max\ d})\)预处理答案,\(O(1)\)输出. 对于\ ...

  5. Unity Profiler 记录

    版本 Unity 2018.4.6f1 空包 development build 魅蓝 note3 OPPO R9 VIVO x9 华为 P8 青春版 小米 8 SE iphone se Other ...

  6. 重构与模式 (Joshua Kerievsky 著)

    第1章 本书的写作缘由 第2章 重构 第3章 模式 第4章 代码坏味 第5章 模式导向的重构目录 第6章 创建 第7章 简化 第8章 泛化 第9章 保护 第10章 聚集操作 第11章 实用重构 参考文 ...

  7. ASP.NET Core快速入门(第2章:配置管理)- 学习笔记(转载)

    原文地址:https://mp.weixin.qq.com/s?__biz=MjM5NjMzMzE2MA==&mid=2451733443&idx=2&sn=6d01721c5 ...

  8. 小记 .NET Core 3.0 下 WPF 是如何运行的

    1. 解决方案架构 如图: 2. 生成的代码 如图: /// <summary> /// App /// </summary> public partial class App ...

  9. C# 流介绍 (原发布 csdn 2017-09-15 23:37:52)

    1.FileStream FileStream 详细介绍参考msdn 写数据: using (FileStream fs = new FileStream("File.FileStream& ...

  10. Java学习——单元测试JUnit

    Java学习——单元测试JUnit 摘要:本文主要介绍了什么是单元测试以及怎么进行单元测试. 部分内容来自以下博客: https://www.cnblogs.com/wxisme/p/4779193. ...