一、嵌套映射
在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站 传送门 真正的干货满满!~