一、myBatis对象关系映射(多对一关系、一对多关系)

1、多对一关系:

---例子:多个员工同属于一个部门。


(1)myBatis发送 额外SQL:

■ 案例:员工表通过 dept_id 关联 部门表,需求:查询指定员工id、name、所属的部门名称的信息。

//部门对象的接口、映射文件省略,跟员工逻辑差不多

/* 员工对象的接口 */
public interface EmployeeMapper {
Employee get(Long id);
} <!--员工对象的映射文件-->
<!-- 解决列名和属性名不匹配问题 -->
<resultMap id="BaseResultMap" type="Employee">
<id column="id" property="id"/>
<result column="name" property="name"/>
<result column="dept_id" property="dept.id"/>
</resultMap> <!-- 查询操作 -->
<select id="get" resultMap="BaseResultMap">
select id, name, dept_id from employee2 where id = #{id}
</select> /* 测试:查询指定员工id、name、所属的部门名称的信息 */
@Test
public void testGet() throws Exception {
SqlSession session = MyBatisUtil.getSession();
EmployeeMapper employeeMapper = session.getMapper(EmployeeMapper.class);
Employee e = employeeMapper.get(1L);
System.out.println(e);
//需要通过查询得到的dept_id查询获取得到部门对象
/**
* 额外的查询语句,可以通过配置resultMap的association属性,让myBatis帮我们执行
*/
//手动添加额外查询语句
// Long dept_id = e.getDept().getId();
// DepartmentMapper departmentMapper = session.getMapper(DepartmentMapper.class);
// Department d = departmentMapper.get(dept_id);
// e.setDept(d); System.out.println(e.getDept());
session.commit();
session.close();
}
  • 上面是通过手动添加的额外SQL查询,通过配置resultMap的association属性,让myBatis帮我们执行如下:

	<!-- 解决列名和属性名不匹配问题 -->
<resultMap id="BaseResultMap" type="Employee">
<id column="id" property="id"/>
<result column="name" property="name"/>
<!--<result column="dept_id" property="dept.id"/> -->
<!-- 额外的SQL配置方式
association元素:配置单一元素的关联关系
property 属性:对象的属性
select 属性:发送额外的sql
column 属性: 将指定的列的值传递给额外sql
-->
<association property="dept"
select="com.shan.hello.mappe r.DepartmentMapper.get"
column="dept_id"
/> </resultMap>

(2)使用额外的 SQL 做映射配置会导致的问题:N+1问题

例如:每个员工的部门编号dept_id 是不同的,当查询所有员工的id、name、所属的部门名称的信息时就会发送额外N条SQL语句。

  • N:发送额外N条SQL语句去查询员工所属的部门名称: select name from department where id = dept_id;

  • 1:查询所有员工的id、name、所属的部门的编号dept_id:select * from employee;

----解决:使用内联映射(多表查询),此时一条 SQL 语句搞定。

----处理多表查询的结果集的方法:内联映射

(3)对象关联的查询:

方式一:额外的SQL

方式二:内联映射

(4)处理多表查询的结果集的方法【内联映射】:

  • 方式一:属性名-列名 通过resultMap的子元素result进行映射配置[内联映射-使用result]:

  • 方式二:属性名-列名 通过resultMap的子元素association进行映射配置[内联映射-使用association]

(5)多对一关系配置总结:

使用association元素,配置单一对象属性。

  • 方式一:额外的SQL[分步查询],一般需要进入另外一个页面展示更加详细的信息。
  • 方式二:内联映射[多表查询],常用。需要在列表中显示关联对象的数据,使用内联映射,否则会出现N+1问题。

2、一对多关系配置:

---例子:一个部门有多个员工。


(1)一对多关系-额外的SQL:

<!-- 针对单一对象的属性,使用association元素 -->
<!-- 针对集合类型的属性,使用collection元素 -->
<!-- 额外的SQL配置方式
collection元素:配置集合类型元素的关联关系
property 属性:对象的属性
select 属性:发送额外的sql
column 属性: 将指定的列的值传递给额外sql
--> <resultMap id="BaseResultMap" type="Department">
<id column="id" property="id"/>
<result column="name" property="name"/>
<!--<result column="" property="emps"/> -->
<collection property="emps"
select="com.shan.hello.mapper.EmployeeMapper.get"
column="id" >
</collection>
</resultMap> <select id="get" resultMap="BaseResultMap">
select id, name from department where id = #{id}
</select>

(2)一对多关系-内联查询:

<!-- 针对单一对象的属性,使用association元素 -->
<!-- 针对集合类型的属性,使用collection元素 -->
<resultMap id="BaseResultMap" type="Department">
<id column="id" property="id"/>
<result column="name" property="name"/>
<!--<result column="" property="emps"/> -->
<!--
内联查询:ofType是集合中泛型的类型
-->
<collection property="emps" ofType="Employee">
<result column="e_id" property="id"/>
<result column="e_name" property="name"/>
<result column="e_dept_id" property="deptId"/>
</collection>
</resultMap> <select id="get" resultMap="BaseResultMap">
<!-- select id, name from department where id = #{id} -->
select d.id, d.name, e.id e_id, e.name e_name, e.dept_id e_dept_id from department d
join employee2 e on d.id = e.dept_id where d.id = #{id}
</select>

二、设置延迟/懒加载

  	<!-- 全局配置文件 -->
<settings>
<!-- 懒加载/延迟加载,开启延迟加载功能 -->
<setting name="lazyLoadingEnabled" value="true"/>
<!-- 设置不要积极地去查询关联对象 -->
<setting name="aggressiveLazyLoading" value="false"/>
<!-- 延迟加载的触发方法 -->
<setting name="lazyLoadTriggerMethods" value="close"/>
</settings>
  • 动态代理-增强功能--延迟加载

总结:多对一、一对多关系的单属性对象/集合属性对象,使用association或collection元素?使用额外SQL或内联查询?

  • 针对【单属性对象】:使用assoication元素,通常需要使用多表查询操作,即内联查询

  • 针对【集合属性对象】:使用collection元素,通常还要使用延迟加载,即额外SQL处理

最新文章

  1. gulp整理
  2. iOS-钥匙串中证书全部失效(证书的签发者无效)的解决办法
  3. {HDU}{2516}{取石子游戏}{斐波那契博弈}
  4. WPF之资源字典zz
  5. hbase基本操作
  6. java_Cookies_1_商品浏览历史记录servlet1
  7. runas /user:administrator cmd 以管理员身份运行CMD
  8. 《Ruby语言入门教程v1.0》学习笔记-03
  9. discuz管理员登录进入后台管理马上跳转到登录界面
  10. Linux System Programming --Chapter Eight
  11. 使用工具intellij idea 进行java web开发简介
  12. 【原创】大叔经验分享(23)spark sql插入表时的文件个数研究
  13. SPRINGMVC中的中文乱码处理
  14. PHP(css样式)
  15. 项目启动一直死循环 DruidDataSource.init 方法
  16. SpringMVC跨重定向请求传递数据
  17. %cd% 与 %~dp0% 区别
  18. kvm安装配置使用centos6.5
  19. Android 混淆代码有关问题总结
  20. USACO 6.5 Checker Challenge

热门文章

  1. Java生成随机数的4种方式
  2. 【安卓】AndroidStudio使用本地gradle进行build的配置
  3. Orthogonal Convolutional Neural Networks
  4. [炼丹术]UNet图像分割模型相关总结
  5. [Error]ValueError: Object arrays cannot be loaded when allow_pickle=False
  6. 基于Spring MVC + Spring + MyBatis的【医院就诊挂号系统】
  7. 云南农职《JavaScript交互式网页设计》 综合机试试卷⑤——简单分类菜单
  8. Java高级程序设计笔记 • 【第2章 多线程(一)】
  9. C/C++ Qt 运用JSON解析库 [基础篇]
  10. linux修改默认的SSH远程端口22