目录
一、MyBatis执行流程
- 读取MyBatis配置文件: mybatis-config.xml加载运行环境和映射文件
- 构造会话工厂SqlSessionFactory
- 会话工厂创建SqlSession对象(包含了执行SQL语句的所有方法)
- 操作数据库的接口,Executor执行器,同时负责查询缓存的维护
- Executor接口的执行方法中有一个MappedStatement类型的参数,封装了映射信息
- 输入参数映射
- 输出结果映射
二、MyBatis是否支持延迟加载?
MyBatis支持延迟加载,默认未开启。
1、什么是延迟加载?
- 查询用户的时候,把用户所属的订单数据也查询出来,这个是立即加载
- 查询用户的时候,暂时不查询订单数据,当需要订单的时候,再查询订单,这个就是延迟加载。
将fetchType设置为lazy
2、延迟加载的原理
- 使用CGLIB创建目标对象的代理对象
- 当调用目标方法user.getOrderList()时,进入拦截器invoke方法,发现user.getOrderLlist()是null值,执行sql查询order列表
- 把order查询上来,然后调用user.setOrderList(List<Order> orderlist),接着完成user.getOrderList()方法的调用
三、MyBatis的缓存
1、一级缓存
基于PerpetualCache的HashMap本地缓存,其存储作用域为Session ,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存。
在同一sqlsession中的相同查询只会执行一次,因为只有第一次的查询结果是存储在一级缓存中的。
2、二级缓存
二级缓存是基于namespace和mapper的作用域起作用的,不是依赖于SQL session,默认也是采用PerpetualCache,HashMap存储,二级缓存默认关闭。
开启二级缓存:
第一步:全局配置文件
第二步:映射文件使用<cache/>标签让**当前mapper(UserMapper)**生效二级缓存
3、注意事项
- 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了新增、修改、删除操作后,默认该作用域下所有select中的缓存将被clear
- 二级缓存需要缓存的数据实现Serializable接口
- 只有会话提交或者关闭以后,一级缓存中的数据才会转移到二级缓存中