mybatis-usage
interview
半ORM框架
- 自己写原生的sql
- 全自动: gorm直接操作对象
User.findone('')
#{}和${}的区别
#{}是预编译处理,PreparedStatement; ${}是字符串替换
dao和xml文件的关系
Dao接口即 Mapper接口。接口的全限名,就是映射文件中mapper.namespace; 接口的方法名,就是映射文件中 Mapper 的 Statement 的 id 值;接口方法内 的参数,就是传递给 sql 的参数;
Mapper接口的工作原理是 JDK 动态代理;生成代理Proxy; 代理的任务是: 代理执行指定的MapperStatement, key=>class.method(namespace+id)
Mapper接口是没有实现类的,当调用接口方法时,接口全限名+方法名拼接字符串作为key值,可唯一定位一个MapperStatement
ResultMap与Java对象的映射关系
- resultMap
- sql列的别名
Mybatis 通过反射创建对象,同时使用反射给对象的属性逐一赋值并返回
动态sql
trim | where | set | foreach | if | choose| when | otherwise | bind
标签
- select|insert|updae|delete
- 、、
- 、
- has-a JAVA对象 bean
- has-a 容器 List
class A{
B b; // association
List<B> bs; //collection
}
class B{}
1对多查询:
<resultMap type="com.lcb.user.Classes" id="ClassesResultMap2">
<id property="id" column="c_id"/>
<result property="name" column="c_name"/>
<association property="teacher" javaType="com.lcb.user.Teacher">
<id property="id" column="t_id"/>
<result property="name" column="t_name"/>
</association>
<collection property="student" ofType="com.lcb.user.Student">
<id property="id" column="s_id"/>
<result property="name" column="s_name"/>
</collection>
</resultMap>
嵌套查询:
column="{subId=id,subName=userName}"; key是子查询中的入参;value是外层查询的结果字段:property;
嵌套查询语句
// 父查询
<resultMap>
<result property="parentId" column="parent_id"/>
<collection property="Java属性名" ofType="Java类名" javaType="ArrayList" column="{subId=parentId}关联子查询" select="sub_select_id(自查询Id)">
</ccollection>
</resultMap>
// 子查询
<select parameterType="int(子查询的入参)" resultType="子查询的Java类名" id="sub_select_id(子查询的id)">
select * from sub where id = #{subId}
<select>
嵌套查询结果
// 父查询
<resultMap>
<result property="parentId" column="parent_id"/>
<collection property="Java属性名" ofType="另一Java类名" javaType="ArrayList" resultMap="sub_result_map(子结果映射)"/>
</resultMap>
// 子结果
<resultMap="sub_result_map" type="子Java类名:subClass">
<id property="id" column=""/>
</resultMap>
延迟加载
Mybatis仅支持association 关联对象和collection关联集合对象的延迟加载,association 指的就是一对一,collection 指的就是一对多查询。
Mybatis配置文件中,可以配置是否启用延迟加载 lazyLoadingEnabled=true|false。
它的原理是,使用 CGLIB 创建目标对象的代理对象,当调用目标方法时,进入 拦截器方法,比如调用 a.getB().getName(),拦截器invoke()方法发现 a.getB()是 null 值,那么就会单独发送事先保存好的查询关联 B 对象的 sql, 把 B 查询上来,然后调用 a.setB(b),于是 a 的对象 b 属性就有值了,接着 完成 a.getB().getName()方法的调用。这就是延迟加载的基本原理。 当然了,不光是 Mybatis,几乎所有的包括 Hibernate,支持延迟加载的原理都 是一样的
二级缓存
- 一级缓存:基于PerpetualCache的HashMap 本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该 Session 中的所有 Cache 就 将清空,
默认打开一级缓存; - 二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap 存储,
不同在于其存储作用域为 Mapper(Namespace),并且可自定义存储源,如 Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实 现 Serializable 序列化接口(可用来保存对象的状态),可在它的映射文件中配 置 ; - 对于缓存数据
更新机制,当某一个作用域(一级缓存 Session/二级缓存 Namespaces)的进行了 C/U/D 操作后,默认该作用域下所有 select 中的缓存将 被 clear
=> cache-aside方法....
接口绑定的实现
- @Select、@Update 等注解
- xml文件
自定义插件
ParameterHandler、ResultSetHandler、 StatementHandler、Executor 这 4 种接口的插件;
通过jdk代理执行InvocationHandler的invoke()方法
- 实现 Mybatis 的Interceptor接口并复写 intercept()方法;
- 然后在给插件编写注解,指定要拦截哪一个接口的哪些方法即可
- 记住,别忘了在配 置文件中配置你编写的插件;
涉及到的设计模式:
调用连模式加载自定义的Interceptorjdk动态代理执行插件:InvocationHandler- 通过
注解匹配接口和方法; => 类似策略;通过注解实现