Mybaits的嵌套映射和动态sql

一、嵌套映射

在Mybatis中,所谓的嵌套映射,就是ResultMap中返回的bean存在其它bean的List的属性,这样就设置到了重复属性的保存,因为关系型数据返回的数据都是二维的,也就是以bean中的List为主,而bean中的属性都是重复的,如下:

这就涉及到了,对这些重复的属性怎么映射的问题,简单的说Mybatis中,就是使用缓存来保存这些属性的bean,发现了重复的属性(根据RowKey(id)),就不在创建bean,下面看下具体实现过程:

入口就是getRowValue(映射, ResultSet)方法。

当读取第一行的时候,首先创建RowKey(默认根据id进行创建,如果没有id,就根据全部映射的result字段进行创建),之后读取缓存数据,当缓存不存在的时候,创建Blog对象同时对属性进行赋值,当存在符合属性的时候,递归操作,当创建对象后进行缓存。

读取第二行的时候,发现已经存在Blog对象,直接使用,然后填充复合属性。

下面进行代码的Debug:

创建RowKey,并根据RowKey中取值:

第一次获取一定是null的:

 所以需要创建对象,同时遍历映射属性:

 这里就是需要填充复合映射的属性:

设置复合属性的MapId:

 获取列明的前缀:

合并子类和父类的RowKey:

获取到值,并且递归调用getRowValue获取复合属性对象的值:

这个时候,对应的复合属性里面,就已经有值了:

 当解析到第三行的时候,暂存区就存在三个值,一个是博客的值,另外一个是两个评论:

 PS:在调试的过程中 columnPrefix="comment_" 增加之后,死活不通过,去掉后可以实现,集体原因不知道为什么。

循环引用的流程

先去把当前对象的时候,放入ancestorObjects中,当在解析复合对象的时候,就会去ancestorObjects寻找是否存在,如果存在则进行关联。

主要就是这段代码:

 ancestorObjects 放入的时候:

 在getRowValue的时候,发现缓存中存在则关联起来:

二、动态sql 

在mybatis中,开发人员相对熟悉的可能就是Mybatis的动态sql了,所谓动态sql,就是可以根据不通的标签,实现不通sql的拼接。首先Mybatis中使用了OGNL表达式对条件进行判断,然后采用解释器模式,对整个动态sql的语法树进行处理,最后形成了需要执行的BoundSql对象。

BoundSql对象的结构如下:

 OGNL表达式简单使用:

Mybatis也是可以支持其他脚本语言,只不过用的比较少了。

脚本的解析流程:把XML -> SqlSource -> BoundSql 的过程。SqlSource存在两种形式,第一种就是动态sql,第二种就是静态sql。

动态脚本的解析流程:

动态Sql最后都会根据标签分解为SqlNode数组,结构如下:

 SqlNode主要使用解释器模式,对每隔标签进行实现:

SqlNode节点的语法树,每执行一个SqlNode节点,都会想DynamicContext中追加相应的Sql语句:

 IF和Where节点的执行:

其实就是数据简单的字符串拼装 ,通过where看到,是继承 TrimSqlNode:

 TrimSqlNode的源码,就是遍历where里面的相关节点,然后截取或者增加相关内容,然后放入一个自有的上下文中,然后进行相关拼装,最后放入整体上下文。

可以看到buffer中是修剪完毕的sql,同时applyAll之后,就是把buffer中的数据,追加到context中:

foreach操作:先根据传入是list还是map,因为循环的时候,需要根据集合内部的数据拼装成参数

#{item_1},#{item_2},如果是map这拼接成#{item_key1},#{item_key2}

 XML的解析过程(xml->SqlSource):

 

 这里推荐mybatis讲解非常棒的福利,这里所有的Mybatis都是跟鲁班大叔学习的:B站 传送门 真正的干货满满!~

 

 

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值